Qt开发技巧(二十二)设置QPA,打开记忆文件,清除表单页注意判断存在性,工程文件去重添加,按钮组的顺序设置,Qt的属性用来传值,查找问题的方法
继续记录一些Qt开发中的技巧操作:
1.设置QPA
平台抽象层,用以取代Qt for Embedded Linux以及Qt4中的平台接口。QPA插件通过定义QPlatform开头的一系列类的子类实现,其中有两个根类,QPlatformIntegration和QPlatformTheme,前者用于窗口系统的集成,后者用于更深层次的平台主题化和集成,QStyle不属于QPA。现在很多linux用wayland作为桌面显示,这样会出现一个问题,由于没有坐标系统,导致无边框窗体无法拖动和定位(一般是Qt6开始强制默认优先用wayland,之前Qt5是默认有xcb则优先用xcb),需要在main函数前面加一行 qputenv(“QT_QPA_PLATFORM”, “xcb”);还有其他选择比如“linuxfb”。看你的Linux嵌入式设备平台的支持情况了。
2.打开记忆文件
有时候导出文件后,希望直接打开文件管理器并选中刚才打开的文件,以便用户打开处理,需要通过执行命令来实现。
QString path = "file:///d:/test1.txt";
QProcess::startDetached("explorer.exe", QStringList() << "/select," << path);
3.清除表单页注意判断存在性
在QTreeWidget/QTableWidget的信号currentItemChanged中,执行对应的clear方法也会触发该信号,这就需要特别注意了,对应该信号的两个参数 current/previous 表示当前节点和上一个节点,两个参数的值都为空,所以在该信号对应槽参数处理中,必须先判断该值是否为空指针,不判断的话很可能导致程序崩溃。
4.工程文件去重添加
在qt的工程文件.pro中,我们常常要添加一些模块,比如“QT += core gui”,“QT+= network websockets”这里的 += 其实可以用 *= 更好,+= 表示添加,不会去重,而 *= 是去重添加,存在则不添加。建议用 *=,尽管 += 也能正常使用,毕竟多一个重复的不影响编译器识别,但这样就存在重复添加的可能。
QT += core gui
QT += core gui
message($$QT) //会打印 core gui core gui
QT *= core gui
QT *= core gui
message($$QT) //会打印 core gui
DEFINES *= abc
DEFINES *= abc
message($$DEFINES) //会打印 abc
5.按钮组的顺序设置
在使用QButtonGroup按钮组添加按钮的时候,如果需要使用buttonClicked(int)信号,则添加按钮的时候必须手动指定按钮编号,否则默认编号都是-1,这样在触发buttonClicked(int)的时候值是负数,导致和预期不一致,原本以为默认按照添加按钮的顺序自动递增设置索引,其实不是的,不会这样处理。务必记得指定按钮编号。
QButtonGroup* btGroup = new QButtonGroup(this);
connect(btnGroup, SIGNAL(buttonClicked(int)), ui->stackedWidget, SLOT(setCurrentIndex(int)));
btGroup->addButton(ui->btWait1, 0);
btGroup->addButton(ui->btWait2, 1);
btGroup->addButton(ui->btWait3, 2;
btGroup->addButton(ui->btWait4, 3);
btGroup->addButton(ui->btWait5, 4);
6.Qt的属性用来传值
Qt的属性机制非常强大,除了可以用来控制样式表,也可以很方便的用来传值,比如qml中的值传递,有时候我们写了一个通用类,希望这个类可以做很多事情,但是又希望其中有一些特殊变量存取值,一种办法是直接定义私有变量,提供get/set接口函数,还有一种偷懒的办法就是用属性setProperty/property,然Qt内部从元对象数据层面自己管理,这样不用在类中写对应的变量和get/set函数。但是肯定有性能损耗,性能上肯定比变量低,所以要看具体的实际需求,如果不是非常频繁的调用setProperty/property,通用性优先的话,那用属性机制会更方便。推荐方式三,继承通用类,在子类中增加set/get。
void MainWindow::on_pushButton_clicked()
{
QElapsedTimer timer;
timer.start();
for (int i = 0; i < 10000; ++i) {
Test *t = new Test;
//t->setId(i);
//t->setName("test");
t->getName();
}
qDebug() << "方式1" << timer.nsecsElapsed();
}
void MainWindow::on_pushButton_2_clicked()
{
QElapsedTimer timer;
timer.start();
for (int i = 0; i < 10000; ++i) {
Test *t = new Test;
//t->setProperty("id", i);
//t->setProperty("name", "test");
t->property("name").toString();
}
qDebug() << "方式2" << timer.nsecsElapsed();
}
7.查找问题的方法
很多时候项目越写越大,然后就可能遇到,明明之前很简单的一段代码,运行的好好的,就那么几行几十行,为何一旦加入到当前项目中,就不行了,百思不得其解。一般遇到这种情况,建议两种处理办法。
办法一:就是注释大法,根据需要,从可能出问题的点,一行一行的注释掉载入的功能模块,甚至有时候你得从main函数入口开始,将不相关的都注释掉,仔细检查运行流程,直到找出引起问题的代码模块。
办法二:就是单独写个最简单的有问题的可以直接编译运行的示例,化繁为简,这样查找问题速度快。往往你会发现,写完这个简单的你怀疑的有问题的代码后,运行是完全正常的,他自己就好了,此时你可以安心的去排查其他代码了。
但有时候,最痛苦的就是简单demo运行的没问题,引入到程序里就不行,就是一些关联模块,互相之间干扰引起,这时候两种方法都得结合着用了。