一份硬核的QT开发经验及资料分享,长文收藏!

[复制链接]
4590|94
 楼主| micoccd 发表于 2022-6-14 11:52 | 显示全部楼层
  • Qt默认不支持大资源文件,比如添加了字体文件,需要pro文件开启。CONFIG += resources_big

  • Qt中继承QWidget之后,样式表不起作用,解决办法有三个。强烈推荐方法一。


  • 方法一:设置属性 this->setAttribute(Qt::WA_StyledBackground, true);
  • 方法二:改成继承QFrame,因为QFrame自带paintEvent函数已做了实现,在使用样式表时会进行解析和绘制。
  • 方法三:重新实现QWidget的paintEvent函数时,使用QStylePainter绘制。
    1. void Widget::paintEvent(QPaintEvent *)
    2. {
    3.     QStyleOption option;
    4.     option.initFrom(this);
    5.     QPainter painter(this);
    6.     style()->drawPrimitive(QStyle::PE_Widget, &option, &painter, this);
    7. }


 楼主| micoccd 发表于 2022-6-14 11:53 | 显示全部楼层
  • 有时候在界面上加了弹簧,需要动态改变弹簧对应的拉伸策略,对应方法为changeSize,很多人会选择使用set开头去找,找不到的。

  • 在使用QFile的过程中,不建议频繁的打开文件写入然后再关闭文件,比如间隔5ms输出日志,IO性能瓶颈很大,这种情况建议先打开文件不要关闭,等待合适的时机比如析构函数中或者日期变了需要重新变换日志文件的时候关闭文件。不然短时间内大量的打开关闭文件会很卡,文件越大越卡。



 楼主| micoccd 发表于 2022-6-14 11:53 | 显示全部楼层
在很多网络应用程序,需要自定义心跳包来保持连接,不然断电或者非法关闭程序,对方识别不到,需要进行超时检测,但是有些程序没有提供心跳协议,此时需要启用系统层的保活程序,此方法适用于TCP连接。
  1. int fd = tcpSocket->socketDescriptor();
  2. int keepAlive = 1;      //开启keepalive属性,缺省值:0(关闭)
  3. int keepIdle = 5;       //如果在5秒内没有任何数据交互,则进行探测,缺省值:7200(s)
  4. int keepInterval = 2;   //探测时发探测包的时间间隔为2秒,缺省值:75(s)
  5. int keepCount = 2;      //探测重试的次数,全部超时则认定连接失效,缺省值:9(次)
  6. setsockopt(fd, SOL_SOCKET, SO_KEEPALIVE, (void *)&keepAlive, sizeof(keepAlive));
  7. setsockopt(fd, SOL_TCP, TCP_KEEPIDLE, (void *)&keepIdle, sizeof(keepIdle));
  8. setsockopt(fd, SOL_TCP, TCP_KEEPINTVL, (void *)&keepInterval, sizeof(keepInterval));
  9. setsockopt(fd, SOL_TCP, TCP_KEEPCNT, (void *)&keepCount, sizeof(keepCount));


 楼主| micoccd 发表于 2022-6-14 11:54 | 显示全部楼层
  • 如果程序打包好以后弹出提示 This application failed to start because it could not find or load the Qt platform plugin 一般都是因为platforms插件目录未打包或者打包错了的原因导致的。

  • 非常不建议tr中包含中文,尽管现在的新版Qt支持中文到其他语言的翻译,但是很不规范,也不知道TMD是谁教的,tr的本意是包含英文,然后翻译到其他语言比如中文,现在大量的初学者滥用tr,如果没有翻译的需求,禁用tr,tr需要开销的,Qt默认会认为他需要翻译,会额外进行特殊处理。

  • 很多人Qt和Qt Creator傻傻分不清楚,经常问Qt什么版本结果发一个Qt Creator的版本过来,Qt Creator是使用Qt编写的集成开发环境IDE,和宇宙第一的Visual Studio一样,他可以是msvc编译器的(WIN对应的Qt集成安装环境中自带的Qt Cerator是msvc的),也可以是mingw编译的,还可以是gcc的。如果是自定义控件插件,需要集成到Qt Creator中,必须保证该插件的动态库文件(dll或者so等文件)对应的编译器和Qt版本以及位数和Qt Creator的版本完全一致才行,否则基本不大可能集成进去。特别注意的是Qt集成环境安装包中的Qt版本和Qt Creator版本未必完全一致,必须擦亮眼睛看清楚,有些是完全一致的。

  • 超过两处相同处理的代码,建议单独写成函数。代码尽量规范精简,比如 if(a == 123) 要写成 if (123 == a),值在前面,再比如 if (ok == true) 要写成 if (ok),if (ok == false) 要写成 if (!ok)等。

  • 很多人问Qt嵌入式平台用哪个好,这里统一回答(当前时间节点2018年):imx6+335x比较稳定,性能高就用RK3288 RK3399,便宜的话就用全志H3,玩一玩可以用树莓派香橙派。



 楼主| micoccd 发表于 2022-6-14 11:55 | 显示全部楼层
  • 对于大段的注释代码,建议用 #if 0 #endif 将代码块包含起来,而不是将该段代码选中然后全部 // ,下次要打开这段代码的话,又需要重新选中一次取消,如果采用的是 #if 0则只要把0改成1即可,效率大大提升。

  • Qt打包发布,有很多办法,Qt5以后提供了打包工具windeployqt(linux上为linuxdeployqt,mac上为macdeployqt)可以很方便的将应用程序打包,使用下来发现也不是万能的,有时候会多打包一些没有依赖的文件,有时候又会忘记打包一些插件尤其是用了qml的情况下,而且不能识别第三方库,比如程序依赖ffmpeg,则对应的库需要自行拷贝,终极**就是将你的可执行文件复制到Qt安装目录下的bin目录,然后整个一起打包,挨个删除不大可能依赖的组件,直到删到正常运行为止。

  • Qt中的动画,底层用的是QElapsedTimer定时器来完成处理,比如产生一些指定规则算法的数据,然后对属性进行处理。



 楼主| micoccd 发表于 2022-6-14 12:57 | 显示全部楼层
在绘制无背景颜色只有边框颜色的圆形时候,可以用绘制360度的圆弧替代,效果完全一致。
  1. QRect rect(-radius, -radius, radius * 2, radius * 2);
  2. //以下两种方法二选一,其实绘制360度的圆弧=绘制无背景的圆形
  3. painter->drawArc(rect, 0, 360 * 16);
  4. painter->drawEllipse(rect);


 楼主| micoccd 发表于 2022-6-14 13:02 | 显示全部楼层
  • 不要把d指针看的很玄乎,其实就是在类的实现文件定义了一个私有类,用来存放局部变量,个人建议在做一些小项目时,没有太大必要引入这种机制,会降低代码可读性,增加复杂性,新手接受项目后会看的很懵逼。

  • 很多人在绘制的时候,设置画笔以为就只可以设置个单调的颜色,其实QPen还可以设置brush,这样灵活性就提高不知道多少倍,比如设置QPen的brush以后,可以使用各种渐变,比如绘制渐变颜色的进度条和文字等,而不再是单调的一种颜色。

  • 很多控件都带有viewport,比如QTextEdit/QTableWidget/QScrollArea,有时候对这些控件直接处理的时候发现不起作用,需要对其viewport()设置才行,比如设置滚动条区域背景透明,需要使用scrollArea->viewport()->setStyleSheet("background-color:transparent;");而不是scrollArea->setStyleSheet("QScrollArea{background-color:transparent;}");

  • 有时候设置了鼠标跟踪setMouseTracking为真,如果该窗体上面还有其他控件,当鼠标移到其他控件上面的时候,父类的鼠标移动事件MouseMove识别不到了,此时需要用到HoverMove事件,需要先设置 setAttribute(Qt::WA_Hover, true);



 楼主| micoccd 发表于 2022-6-14 13:05 | 显示全部楼层
Qt封装的QDateTime日期时间类非常强大,可以字符串和日期时间相互转换,也可以毫秒数和日期时间相互转换,还可以1970经过的秒数和日期时间相互转换等。
  1. QDateTime dateTime;
  2. QString dateTime_str = dateTime.currentDateTime().toString("yyyy-MM-dd hh:mm:ss");
  3. //从字符串转换为毫秒(需完整的年月日时分秒)
  4. datetime.fromString("2011-09-10 12:07:50:541", "yyyy-MM-dd hh:mm:ss:zzz").toMSecsSinceEpoch();
  5. //从字符串转换为秒(需完整的年月日时分秒)
  6. datetime.fromString("2011-09-10 12:07:50:541", "yyyy-MM-dd hh:mm:ss:zzz").toTime_t();
  7. //从毫秒转换到年月日时分秒
  8. datetime.fromMSecsSinceEpoch(1315193829218).toString("yyyy-MM-dd hh:mm:ss:zzz");
  9. //从秒转换到年月日时分秒(若有zzz,则为000)
  10. datetime.fromTime_t(1315193829).toString("yyyy-MM-dd hh:mm:ss[:zzz]");


 楼主| micoccd 发表于 2022-6-14 13:11 | 显示全部楼层
在我们使用QList、QStringList、QByteArray等链表或者数组的过程中,如果只需要取值,而不是赋值,强烈建议使用 at() 取值而不是 [] 操作符,在官方书籍《C++ GUI Qt 4编程(第二版)》的书中有特别的强调说明,此教材的原作者据说是Qt开发的核心人员编写的,所以还是比较权威,至于使用 at() 与使用 [] 操作符速度效率的比较,网上也有网友做过此类对比。原文在书的212页,这样描述的:Qt对所有的容器和许多其他类都使用隐含共享,隐含共享是Qt对不希望修改的数据决不进行复制的保证,为了使隐含共享的作用发挥得最好,可以采用两个新的编程习惯。第一种习惯是对于一个(非常量的)向量或者列表进行只读存取时,使用 at() 函数而不用 [] 操作符,因为Qt的容器类不能辨别 [] 操作符是否将出现在一个赋值的左边还是右边,他假设最坏的情况出现并且强制执行深层赋值,而 at() 函数则不被允许出现在一个赋值的左边。
 楼主| micoccd 发表于 2022-6-14 13:13 | 显示全部楼层
如果是dialog窗体,需要在exec以后还能让其他代码继续执行,请在dialog窗体exec前增加一行代码,否则会阻塞窗体消息。
  1. QDialog dialog;
  2. dialog.setWindowModality(Qt::WindowModal);
  3. dialog.exec();
 楼主| micoccd 发表于 2022-6-14 13:13 | 显示全部楼层
  • 安全的删除Qt的对象类,强烈建议使用deleteLater而不是delete,因为deleteLater会选择在合适的时机进行释放,而delete会立即释放,很可能会出错崩溃。如果要批量删除对象集合,可以用qDeleteAll,比如 qDeleteAll(btns);

  • 在QTableView控件中,如果需要自定义的列按钮、复选框、下拉框等其他模式显示,可以采用自定义委托QItemDelegate来实现,如果需要禁用某列,则在自定义委托的重载createEditor函数返回0即可。自定义委托对应的控件在进入编辑状态的时候出现,如果想一直出现,则需要重载paint函数用drawPrimitive或者drawControl来绘制。

  • 将 QApplication::style() 对应的drawPrimitive、drawControl、drawItemText、drawItemPixmap等几个方法用熟悉了,再结合QStyleOption属性,可以玩转各种自定义委托,还可以直接使用paint函数中的painter进行各种绘制,各种牛X的表格、树状列表、下拉框等,绝对屌炸天。QApplication::style()->drawControl 的第4个参数如果不设置,则绘制出来的控件不会应用样式表。

  • 心中有坐标,万物皆painter,强烈建议在学习自定义控件绘制的时候,将qpainter.h头文件中的函数全部看一遍、试一遍、理解一遍,这里边包含了所有Qt内置的绘制的接口,对应的参数都试一遍,你会发现很多新大陆,会大大激发你的绘制的兴趣,犹如神笔马良一般,策马崩腾遨游代码绘制的世界。



 楼主| micoccd 发表于 2022-6-14 13:15 | 显示全部楼层
在使用setItemWidget或者setCellWidget的过程中,有时候会发现设置的控件没有居中显示而是默认的左对齐,而且不会自动拉伸填充,对于追求完美的程序员来说,这个可不大好看,有个终极通用办法就是,将这个控件放到一个widget的布局中,然后将widget添加到item中,这样就完美解决了,而且这样可以组合多个控件产生复杂的控件。
  1. //实例化进度条控件
  2. QProgressBar *progress = new QProgressBar;
  3. //增加widget+布局巧妙实现居中
  4. QWidget *widget = new QWidget;
  5. QHBoxLayout *layout = new QHBoxLayout;
  6. layout->setSpacing(0);
  7. layout->setMargin(0);
  8. layout->addWidget(progress);
  9. widget->setLayout(layout);
  10. ui->tableWidget->setCellWidget(0, 0, widget);


 楼主| micoccd 发表于 2022-6-14 13:15 | 显示全部楼层
很多时候需要在已知背景色的情况下,能够清晰的绘制文字,这个时候需要计算对应的文字颜色。
  1. //根据背景色自动计算合适的前景色
  2. double gray = (0.299 * color.red() + 0.587 * color.green() + 0.114 * color.blue()) / 255;
  3. QColor textColor = gray > 0.5 ? Qt::black : Qt::white;
 楼主| micoccd 发表于 2022-6-14 13:16 | 显示全部楼层
对QTableView或者QTableWidget禁用列拖动。
  1. #if (QT_VERSION <= QT_VERSION_CHECK(5,0,0))
  2.     ui->tableView->horizontalHeader()->setResizeMode(0, QHeaderView::Fixed);
  3. #else
  4.     ui->tableView->horizontalHeader()->setSectionResizeMode(0, QHeaderView::Fixed);
  5. #endif
 楼主| micoccd 发表于 2022-6-14 13:17 | 显示全部楼层
从Qt4转到Qt5,有些类的方法已经废弃或者过时了,如果想要在Qt5中启用Qt4的方法,比如QHeadVew的setMovable,可以在你的pro或者pri文件中加上一行即可:DEFINES += QT_DISABLE_DEPRECATED_BEFORE=0

Qt中的QColor对颜色封装的很完美,支持各种转换,比如rgb、hsb、cmy、hsl,对应的是toRgb、toHsv、toCmyk、toHsl,还支持透明度设置,颜色值还能转成16进制格式显示。
  1. QColor color(255, 0, 0, 100);
  2. qDebug() << color.name() << color.name(QColor::HexArgb);
  3. //输出 #ff0000 #64ff0000
 楼主| micoccd 发表于 2022-6-14 13:18 | 显示全部楼层
QVariant类型异常的强大,可以说是万能的类型,在进行配置文件的存储的时候,经常会用到QVariant的转换,QVariant默认自带了toString、toFloat等各种转换,但是还是不够,比如有时候需要从QVariant转到QColor,而却没有提供toColor的函数,这个时候就要用到万能办法。
  1. if (variant.typeName() == "QColor") {
  2.     QColor color = variant.value<QColor>();
  3.     QFont font = variant.value<QFont>();
  4.     QString nodeValue = color.name(QColor::HexArgb);
  5. }
 楼主| micoccd 发表于 2022-6-14 13:18 | 显示全部楼层
  • Qt中的QString和const char *之间转换,最好用toStdString().c_str()而不是toLocal8Bit().constData(),比如在setProperty中如果用后者,字符串中文就会不正确,英文正常。

  • Qt的信号槽机制非常牛X,也是Qt的独特的核心功能之一,有时候我们在很多窗体中传递信号来实现更新或者处理,如果窗体层级比较多,比如窗体A的父类是窗体B,窗体B的父类是窗体C,窗体C有个子窗体D,如果窗体A一个信号要传递给窗体D,问题来了,必须先经过窗体B中转到窗体C再到窗体D才行,这样的话各种信号关联信号的connect会非常多而且管理起来比较乱,可以考虑增加一个全局的单例类AppEvent,公共的信号放这里,然后窗体A对应信号绑定到AppEvent,窗体D绑定AppEvent的信号到对应的槽函数即可,干净清爽整洁。

  • QTextEdit右键菜单默认英文的,如果想要中文显示,加载widgets.qm文件即可,一个Qt程序中可以安装多个翻译文件,不冲突。

  • Qt中有个全局的焦点切换信号focusChanged,可以用它做自定义的输入法。Qt4中默认会安装输入法上下文,比如在main函数打印a.inputContext会显示值,这个默认安装的输入法上下文,会拦截两个牛X的信号QEvent::RequestSoftwareInputPanel和QEvent::CloseSoftwareInputPanel,以至于就算你安装了全局的事件过滤器依然识别不到这两个信号,你只需要在main函数执行a.setInputContext(0)即可,意思是安装输入法上下文为空。

  • 在Qt5.10以后,表格控件QTableWidget或者QTableView的默认最小列宽改成了15,以前的版本是0,所以在新版的qt中,如果设置表格的列宽过小,不会应用,取的是最小的列宽。所以如果要设置更小的列宽需要重新设置ui->tableView->horizontalHeader()->setMinimumSectionSize(0);



 楼主| micoccd 发表于 2022-6-14 13:28 | 显示全部楼层
Qt源码中内置了一些未公开的不能直接使用的黑科技,都藏在对应模块的private中,比如gui-private widgets-private等,比如zip文件解压类QZipReader、压缩类QZipWriter就在gui-private模块中,需要在pro中引入QT += gui-private才能使用。
  1. #include "QtGui/private/qzipreader_p.h"
  2. #include "QtGui/private/qzipwriter_p.h"

  3. QZipReader reader(dirPath);
  4. QString path("");
  5. //解压文件夹到当前目录
  6. reader.extractAll(path);
  7. //文件夹名称
  8. QZipReader::FileInfo fileInfo = reader.entryInfoAt(0);
  9. //解压文件
  10. QFile file(filePath);
  11. file.open(QIODevice::WriteOnly);
  12. file.write(reader.fileData(QString::fromLocal8Bit("%1").arg(filePath)));
  13. file.close();
  14. reader.close();

  15. QZipWriter *writer = new QZipWriter(dirPath);
  16. //添加文件夹
  17. writer->addDirectory(unCompress);
  18. //添加文件
  19. QFile file(filePath);
  20. file.open(QIODevice::ReadOnly);
  21. writer->addFile(data, file.readAll());
  22. file.close();
  23. writer->close();


 楼主| micoccd 发表于 2022-6-14 13:29 | 显示全部楼层
  • 理论上串口和网络收发数据都是默认异步的,操作系统自动调度,完全不会卡住界面,网上那些说收发数据卡住界面主线程的都是扯几把蛋,真正的耗时是在运算以及运算后的处理,而不是收发数据,在一些小数据量运算处理的项目中,一般不建议动用线程去处理,线程需要调度开销的,不要什么东西都往线程里边扔,线程不是万能的。只有当真正需要将一些很耗时的操作比如编码解码等,才需要移到线程处理。

  • 在构造函数中获取控件的宽高很可能是不正确的,需要在控件首次显示以后再获取才是正确的,控件是在首次显示以后才会设置好正确的宽高值,记住是在首次显示以后,而不是构造函数或者程序启动好以后,如果程序启动好以后有些容器控件比如QTabWidget中的没有显示的页面的控件,你去获取宽高很可能也是不正确的,万无一失的办法就是首次显示以后去获取。

  • 数据库处理一般建议在主线程,如果非要在其他线程,务必记得打开数据库也要在那个线程,即在那个线程使用数据库就在那个线程打开,不能打开数据库在主线程,执行sql在子线程,很可能出问题。



 楼主| micoccd 发表于 2022-6-14 13:30 | 显示全部楼层
新版的QTcpServer类在64位版本的Qt下很可能不会进入incomingConnection函数,那是因为Qt5对应的incomingConnection函数参数变了,由之前的int改成了qintptr,改成qintptr有个好处,在32位上自动是quint32而在64位上自动是quint64,如果在Qt5中继续写的参数是int则在32位上没有问题在64位上才有问题,所以为了兼容Qt4和Qt5,必须按照不一样的参数写。
  1. #if (QT_VERSION > QT_VERSION_CHECK(5,0,0))
  2.     void incomingConnection(qintptr handle);
  3. #else
  4.     void incomingConnection(int handle);
  5. #endif


您需要登录后才可以回帖 登录 | 注册

本版积分规则

快速回复 在线客服 返回列表 返回顶部