0

我正在编写一个 qt 桥以通过 Web 应用程序访问串行热敏打印机。我在设置串行端口设置时遇到问题,这是我的代码:

bool Bridge::printToSerial(QByteArray arr, QString serialPortName, quint16 baud) {

/*
QProcess    proc;
QStringList args = QStringList() << QString("mode COM1 BAUD=%1 PARITY=%2 DATA=%3" ).arg(9600).arg( "n" ).arg(8);
proc.start( "cmd.exe", args );
proc.waitForStarted();
proc.close();
*/

QSerialPort* m_port = new QSerialPort(this);
m_port->setPortName(serialPortName); // \\.\COM1
if(!m_port->open(QSerialPort::ReadWrite)) {
   qDebug() << "Error serial open";
} else {
    if (arr.isEmpty()) {
        qDebug() << QObject::tr("Either no data was currently available on the standard input for reading, or an error occurred for port %1, error: %2").arg(serialPortName).arg(m_port->errorString()) << endl;
        delete m_port;
        return 1;
    }

    m_port->setBaudRate(baud);
    m_port->setDataBits(QSerialPort::Data8);
    m_port->setParity(QSerialPort::NoParity);
    m_port->setStopBits(QSerialPort::OneStop);

    //m_port->setBreakEnabled(true);

    qint64 bytesWritten = m_port->write(arr);
    if (bytesWritten == -1) {
        qDebug() << QObject::tr("Failed to write the data to port %1, error: %2").arg(serialPortName).arg(m_port->errorString()) << endl;
        delete m_port;
        return 1;
    } else if (bytesWritten != arr.size()) {
        qDebug() << QObject::tr("Failed to write all the data to port %1, error: %2").arg(serialPortName).arg(m_port->errorString()) << endl;
        delete m_port;
        return 1;
    } else if (!m_port->waitForBytesWritten(500)) {
        qDebug() << QObject::tr("Operation timed out or an error occurred for port %1, error: %2").arg(serialPortName).arg(m_port->errorString()) << endl;
        delete m_port;
        return 1;
    }
    m_port->close();
    delete m_port;
    qDebug() << QObject::tr("Data successfully sent to port %1").arg(serialPortName) << endl;
    return false;
}
delete m_port;
return true;
}

问题是数据打印但没有完成,这让我觉得数据位没有设置为 8 位。

我执行了命令:mode COM1,它显示端口配置不正确(波特率、位、奇偶校验都错误)。顺便说一句,设备管理器显示端口的默认设置和与 cmd 行不同的结果:“mode COM1”

为了使它工作,我必须在运行程序之前执行 cmd 行:mode COM1 BAUD=9600 PARITY=n DATA=8。或调用 m_port->setBreakEnabled(true); 在我打开端口之后。此调用挂起程序,我必须重新编译 exe 而无法工作。当系统重新启动时,命令行模式显示旧结果(波特率 1200,数据位 7 等)

任何线索为什么我有这种行为?似乎 QSerialPort 无法有效地更改我需要的所有设置。

4

2 回答 2

1

为什么每次发送一些数据时都要打开和关闭端口?此外,仅仅因为writeorwaitForBytesWritten已经返回表明数据被“写入”的东西,它只意味着数据已经从顶级设备驱动程序的缓冲区中刷新。将所有数据实际推送到硬件的 USB 请求可能仍处于未决状态,设备的 FIFO 可能仍有数据,等等。

然而,当您关闭设备时,您通常会中止任何仍在进行中的事务,并导致硬件清除其缓冲区。写入是异步和流式传输以保持性能,设备关闭是一种同步的即时结果类型的操作。

保持设备打开。仅当您以特定于平台的方式获知设备将要离开或添加新设备时,才需要重新打开它。

于 2016-07-19T18:33:25.993 回答
0

@Kuba Ober 是对的。我最终将函数修改为如下所示,跟踪打开的端口:(我不知道是否应该为每个端口设置一个 QSerialPort 指针,或者如下所示,每次写入数据时使用它来重新定义端口设置,但上述方法仍然有效。)

bool Bridge::printToSerial(QByteArray arr, QString serialPortName, quint16 baud) {

if (arr.isEmpty()) {
    qDebug() << QObject::tr("Either no data was currently available on the standard input for reading, or an error occurred for port %1, error: %2").arg(serialPortName).arg(m_port->errorString()) << endl;
    return false;
}

bool portOpen = m_opened.contains(serialPortName);

if(!portOpen) {
    if(m_port == NULL) {
        m_port = new QSerialPort(this);
    }
    m_port->setPortName(serialPortName);
    if(!m_port->open(QSerialPort::WriteOnly)) {
        qDebug() << "Error serial open";
        return false;
    }
    portOpen = true;
    m_opened[serialPortName] = true;
} else {
    m_port->setPortName(serialPortName); // \\.\COM1
}

m_port->setBaudRate(baud);
m_port->setDataBits(QSerialPort::Data8);
m_port->setParity(QSerialPort::NoParity);
m_port->setStopBits(QSerialPort::OneStop);

qint64 bytesWritten = m_port->write(arr);
if (bytesWritten == -1) {
    qDebug() << QObject::tr("Failed to write the data to port %1, error: %2").arg(serialPortName).arg(m_port->errorString()) << endl;
    delete m_port;
    return 1;
} else if (bytesWritten != arr.size()) {
    qDebug() << QObject::tr("Failed to write all the data to port %1, error: %2").arg(serialPortName).arg(m_port->errorString()) << endl;
    delete m_port;
    return 1;
} else if (!m_port->waitForBytesWritten(5000)) {
    qDebug() << QObject::tr("Operation timed out or an error occurred for port %1, error: %2").arg(serialPortName).arg(m_port->errorString()) << endl;
    delete m_port;
    return 1;
}
qDebug() << QObject::tr("Data successfully sent to port %1").arg(serialPortName) << endl;
return true;
}
于 2016-07-19T19:41:40.747 回答