这个基于Qt5.12开发的串口调试助手是我在实际工作中经常使用的一个实用工具。它虽然界面简洁,但功能非常实用,特别适合嵌入式开发、硬件调试等场景。作为一个经常需要和单片机、传感器打交道的开发者,一个好用的串口调试工具能极大提升工作效率。
这个工具最吸引我的地方在于它完整实现了串口通信的核心功能,包括:
对于初学者来说,这是一个很好的学习Qt串口编程的范例;对于有经验的开发者,它也是一个可以快速上手的调试工具。我在实际使用过程中还根据自己的需求做了一些功能扩展,后面会详细分享这些经验。
要运行和开发这个项目,你需要准备以下环境:
提示:建议将Qt安装路径和项目路径都设置为纯英文路径,避免可能出现的编码问题。
在Qt Creator中打开项目后,有几个关键配置需要注意:
qmake复制QT += core gui serialport
greaterThan(QT_MAJOR_VERSION, 4): QT += widgets
TARGET = SerialDebugTool
TEMPLATE = app
bash复制sudo usermod -a -G dialout $USER
cpp复制QTextCodec::setCodecForLocale(QTextCodec::codecForName("UTF-8"));
串口的初始化是整个工具的基础,在SerialPort类中实现。以下是关键代码分析:
cpp复制bool SerialPort::openPort(const QString &portName)
{
if(port->isOpen()) {
port->close();
}
port->setPortName(portName);
if(!port->open(QIODevice::ReadWrite)) {
qDebug() << "Failed to open port" << portName;
return false;
}
// 默认配置
port->setBaudRate(QSerialPort::Baud9600);
port->setDataBits(QSerialPort::Data8);
port->setParity(QSerialPort::NoParity);
port->setStopBits(QSerialPort::OneStop);
port->setFlowControl(QSerialPort::NoFlowControl);
return true;
}
在实际使用中,我发现以下几个参数需要特别注意:
发送功能支持文本和十六进制两种模式:
cpp复制void MainWindow::on_sendButton_clicked()
{
QString text = ui->sendTextEdit->toPlainText();
if(text.isEmpty()) return;
QByteArray data;
if(ui->hexSendCheckBox->isChecked()) {
// 十六进制发送处理
data = QByteArray::fromHex(text.toLatin1());
} else {
// 文本模式发送
data = text.toUtf8();
}
if(serialPort->write(data) == -1) {
statusBar()->showMessage("发送失败", 2000);
} else {
statusBar()->showMessage("发送成功", 2000);
}
}
接收处理通过readyRead信号触发:
cpp复制void SerialPort::handleReadyRead()
{
QByteArray data = port->readAll();
// 处理接收到的数据
if(!data.isEmpty()) {
emit dataReceived(data);
}
}
在MainWindow中,接收到的数据会根据设置以文本或十六进制形式显示:
cpp复制void MainWindow::onDataReceived(const QByteArray &data)
{
QString displayText;
if(ui->hexDisplayCheckBox->isChecked()) {
// 十六进制显示
displayText = data.toHex(' ').toUpper();
} else {
// 文本显示
displayText = QString::fromUtf8(data);
}
ui->receiveTextEdit->append(displayText);
// 自动回环处理
if(ui->loopbackCheckBox->isChecked()) {
on_sendButton_clicked();
}
}
自动回环是调试时非常实用的功能,实现原理很简单:
cpp复制void MainWindow::on_loopbackCheckBox_stateChanged(int state)
{
loopbackEnabled = (state == Qt::Checked);
}
void MainWindow::onDataReceived(const QByteArray &data)
{
// ...显示接收数据代码...
if(loopbackEnabled) {
// 直接将接收到的数据放入发送框并触发发送
ui->sendTextEdit->setPlainText(
ui->hexDisplayCheckBox->isChecked()
? data.toHex(' ')
: QString::fromUtf8(data)
);
on_sendButton_clicked();
}
}
在实际使用中,我对原始代码做了一些功能增强,这些改进让工具更加实用:
cpp复制void MainWindow::saveSendHistory()
{
QString text = ui->sendTextEdit->toPlainText();
if(!text.isEmpty() && !sendHistory.contains(text)) {
sendHistory.append(text);
if(sendHistory.size() > 10) {
sendHistory.removeFirst();
}
updateHistoryMenu();
}
}
void MainWindow::updateHistoryMenu()
{
QMenu *historyMenu = new QMenu(this);
for(const QString &item : sendHistory) {
QAction *action = historyMenu->addAction(item);
connect(action, &QAction::triggered, [this, item](){
ui->sendTextEdit->setPlainText(item);
});
}
ui->sendButton->setMenu(historyMenu);
}
cpp复制void MainWindow::on_timerSendCheckBox_stateChanged(int state)
{
if(state == Qt::Checked) {
sendTimerId = startTimer(ui->timerIntervalSpinBox->value());
} else {
killTimer(sendTimerId);
}
}
void MainWindow::timerEvent(QTimerEvent *event)
{
if(event->timerId() == sendTimerId) {
on_sendButton_clicked();
}
}
cpp复制void MainWindow::updateCounters(const QByteArray &data, bool isSend)
{
if(isSend) {
sendBytes += data.size();
sendPackets++;
ui->sendCountLabel->setText(
QString("发送: %1 字节 / %2 包").arg(sendBytes).arg(sendPackets));
} else {
receiveBytes += data.size();
receivePackets++;
ui->receiveCountLabel->setText(
QString("接收: %1 字节 / %2 包").arg(receiveBytes).arg(receivePackets));
}
}
可能原因及解决方法:
串口被其他程序占用
lsof /dev/tty*权限问题(Linux/macOS)
bash复制sudo usermod -a -G dialout $USER
sudo chmod 666 /dev/ttyS*
串口名称错误
常见问题排查步骤:
解决方案:
cpp复制QTextCodec::setCodecForLocale(QTextCodec::codecForName("UTF-8"));
当处理大量数据时,直接追加到QTextEdit会影响性能。可以优化为:
cpp复制void MainWindow::onDataReceived(const QByteArray &data)
{
static QString buffer;
// 处理显示数据...
buffer.append(displayText);
// 每100ms更新一次显示
if(!updateTimer->isActive()) {
updateTimer->start(100);
}
}
void MainWindow::updateDisplay()
{
ui->receiveTextEdit->append(buffer);
buffer.clear();
updateTimer->stop();
}
对于高速串口通信,可以考虑将串口操作放在单独的线程中:
cpp复制class SerialThread : public QThread
{
Q_OBJECT
public:
explicit SerialThread(QObject *parent = nullptr);
void run() override;
void writeData(const QByteArray &data);
signals:
void dataReceived(const QByteArray &data);
private:
QSerialPort *port;
QMutex mutex;
QQueue<QByteArray> writeQueue;
};
实现一个环形缓冲区来缓存接收数据:
cpp复制class RingBuffer
{
public:
RingBuffer(int size = 4096);
bool write(const QByteArray &data);
QByteArray read(int maxSize);
int available() const;
private:
QByteArray buffer;
int head = 0;
int tail = 0;
const int capacity;
};
在与STM32系列单片机通信时,我通常会这样配置:
典型的调试流程:
这个工具也可以用于简单的Modbus协议调试。例如发送Modbus RTU查询指令:
code复制01 03 00 00 00 01 84 0A
解释:
以常见的温湿度传感器为例,接收到的数据可能是这样的:
code复制55 01 01 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00