QT中文乱码
QTextCodec::setCodecForCStrings(QTextCodec::codecForName("GB2312"));
通过signal/slot来传递自定义类型的值
•自定一种类型,在这个类型的顶部包含:#include <QMetaType>
•在类型定义完成后,加入声明:Q_DECLARE_METATYPE(TextAndNumber);
•在main()函数中注册这种类型:qRegisterMetaType<TextAndNumber>("TextAndNumber");
•如果还希望使用这种类型的引用,可同样要注册:qRegisterMetaType<TextAndNumber>("TextAndNumber&");
QString 转换为 char*
QByteArray ba = src.toLatin1();
const char s = ba.data();
或者
char ch;
std::string str = m_ip.toStdString();
ch = (char )malloc((str.length()+1)sizeof(char));
memset(ch,0,str.length()+1);
str.copy(ch,str.length(),0);
或者
ch = (char )malloc((str.length()+1)sizeof(char));
str.copy(ch,str.length(),0);
ch[str.length()] = '\0';
不调用connect自动连接槽
如果槽的命名是这样的话:void on_<widget name><signal name>(<signal parameters>);
就会自动将widget name中的信号signal name和这个槽void on<widget name>_<signal name>(<signal parameters>)链接起来。
这是QT不需要connect语句就可以自动链接信号和槽的机制!
QThread线程
在次线程中使用QT的类需要注意:
- QObject的子对象必须在它的父对象线程中创建。
- 在删除对应的QThread对象之前,必须删除所有在次线程中创建的QObject对象
- 必须在创建QObject对象的线程中删除它们
QTimer问题
QTimer使用信号/槽的时候必须在创建QTimer的那个线程中的线程执行exec()方法。但是执行exec()后线程进入事件循环,只有执行了exit()或quit()后才退出。
所以个人认为:QTimer只能在GUI的主线程中创建,因为主线程会执行exec()后进入事件循环。
若QTimer在子线程中创建,子线程不执行exec(),QTimer永远不会发送timeout()信号。
qDebug
%a,%A 读入一个浮点值(仅C99有效)
%c 读入一个字符
%d 读入十进制整数
%i 读入十进制,八进制,十六进制整数
%o 读入八进制整数
%x,%X 读入十六进制整数
%s 读入一个字符串,遇空格、制表符或换行符结束。
%f,%F,%e,%E,%g,%G 用来输入实数,可以用小数形式或指数形式输入。
%p 读入一个指针
%u 读入一个无符号十进制整数
%n 至此已读入值的等价字符数
%[] 扫描字符集合
%% 读%符号
qss用法
背景图片九宫格
border-width: 5px;
border-image: url(./run/images/news/group_normal.png) 5 5 5 5 stretch stretch;
border-width必须要加上,用border-image来自动根据大小扩展,而不用background-image
字体颜色
color: white;
按钮3种状态:正常,鼠标置上,按下
QPushButton
QPushButton:hover
QPushButton:checked
对于checked,要把QPushButton设置为可checkable。(setCheckable(true))
参考:http://www.cppblog.com/seahouse/archive/2011/03/29/142730.html
设置QPushButton鼠标指向时变成手形
pushButton->setCursor(QCursor(Qt::PointingHandCursor))
Qt窗口操作函数(最大化,全屏,隐藏最大化,最小化)
//Qt主窗口没有最小化,最大化按钮且最大化显示
int main(int argc, char *argv[])
{
QApplication a(argc, argv);
TestQtForWinCE w;
w.setWindowFlags(w.windowFlags()& ~Qt::WindowMaximizeButtonHint& ~Qt::WindowMinimizeButtonHint);
w.showMaximized();
return a.exec();
}
这里的“&~”是取反以后再按位与的意思,下面的“|”是按位或的意思
//同时禁止最大化最小化按钮
Qt::WindowMinMaxButtonsHint
//也禁止关闭
w.setWindowFlags(w.windowFlags() &~ (Qt::WindowMinMaxButtonsHint | Qt::WindowCloseButtonHint));
Qt全屏显示函数
1、window.showFullScreen()//此方法只对顶级窗口有效,对子窗口无效
2、yourwidget->setWindowFlags(Qt::window | Qt::FramelessWindowHint);
(第一个Qt::window表示此widget是窗口类型,第二个参数使用无框架就是没有标题,状态栏和边框)
Qt最大化显示函数 window.showMaximized()
Qt最小化显示函数 window.showMinimized()
Qt固定尺寸显示函数 window.resize(x,y)
子窗口全屏显示:
QT中窗口部件QWidget成员函数showFullScreen();是用于将窗口部件全屏显示,但是他只对窗口模式的部件有用。子窗口的特征是 Qt::SubWindow,不是独立的窗口。因此对其调用showFullScreen()无效。通过对对子窗口调 用:setWindowFlags(Qt::Dialog);或setWindowFlags(Qt::Window);将其设为窗口模式后,即可调用 showFullScreen();进行全屏显示了。
如果要还原为普通窗口,调用setWindowFlags(Qt::subwindow)将子窗口设置为非顶级窗口,再调用showNormal()还原子窗口显示。
参考:
相对于子窗口的全屏显示方法,不得不提的是子窗口退出全屏的方法。
MFC中,子窗口退出全屏的方法简单直接,直接调用SetWindowPos()函数将子窗口显示到预定的位置即可。
QT中,直接使用move和resize对窗口定位是没有效果的。需要先将子窗口设位非窗口模式。之后再将窗口移到理想位置。
即先调用setWindowFlags(Qt::Dialog);或者setWindowFlags(Qt::SubWindow);在调用move和resize函数。
有人提到退出全屏时,调用showNormal(),再调setWindowFlags(Qt::SubWindow)。
这有两个问题:1)showNormal()也是只对顶级窗口有效。2)可以考虑先调用showNormal()将窗口设为原来大小,在将窗口设为子窗口模式。但这样窗口回到父窗口之中也不会显示原来的大小。
获取屏幕宽度和高度
QApplication::desktop()->width() ;
QApplication::desktop()->height() ;
即得到屏幕分辨率,如1024*768,800*600
QSS样式
Qt的CSS帮助文档
打开Qt Assistant工具,在索引页面查找关键字:
1、“Qt Style Sheets Reference”查看Qt样式表的用法
2、“Qt Style Sheets Examples” 查看Qt样式表的实例
3、“The Style Sheet Syntax” 查看Qt样式表语法
QT utf8乱码
xml是utf8格式,读取的中文调用如下方法:
QString::fromUtf8(pTagChildElement->GetText()));
QT 子线程不能调用主窗口线程的控件
子线程不能调用主窗口线程的控件,否则会出现莫名其妙的错误,
解决办法,通过信号/槽机制解决
QObject::connect: Cannot queue arguments of type 'MediaDbAttirbute'(Make sure 'MediaDbAttirbute' is registered using qRegisterMetaType().)
信号/槽的参数需要注册为QT的元类型,解决方法:
qRegisterMetaType<MediaDbAttirbute>("MediaDbAttirbute");
QT QListWidget删除item问题
只调用removeItemWidget,不销毁指针指向的内存,会无效果。
QListWidgetItem* pItem = ui.msgListWidget->item(100);
ui.msgListWidget->removeItemWidget(pItem);
delete pItem;
pItem = NULL;
QT QMap问题
当使用QMap::iterator时,同一个iter如果map改变后,需要重新获取该map的iterator;
原来的Iter还记录的未更新前的数组映射表,如果这时候引用老的iter,map的count变短后,可能导致队列溢出
QProcess不能实时标准输出
在被调用的进程里的main函数中添加如下代码:
setvbuf(stdout, (char *)NULL, _IONBF, 0);
QMutex与QMutexLocker互斥锁问题
区别:
- QMutex需要手动lock和unlock, 如果函数有多个出口,可能会死锁
- QMutexLocker不需要unlock,在代码块结束时候自动释放
- QMutex对象被锁住以后,其他调用该锁的代码块均不能访问,直到锁被解开
QT信号和槽
- 需要在线程队列存在的情况下才能执行,需要使线程进入监听状态,执行exec()函数
- 线程间通信,线程需要调用moveToThread(this);