Qt刷新机制的一些总结(Qt内部画的时候是相当于画在后台一个对象里,然后在刷新的时候调用bitblt统一画,调用window的api并不会影响到后面的那个对象)
2016-10-10 22:50
447 查看
前段时间做过一个界面刷新的优化,遇到的坑比较多,在这里做一点点总结吧。
优化的方案是滚动滚动条的时候用截屏的方式代替界面全部刷新,优化完成后,界面在滚动时效率能提升大概一倍,背景介绍完毕。
用到最主要的是QT的截屏功能
window原生api会提供截屏滚动的功能。可以用这个ScrollWindowEx这个api。它会根据相应的参数在屏幕中进行滚动相应的区域。是不是很容易?但是结果却是不理想,因为用的是Qt,控件重写过PaintEvent的方法。调用api实时能看到效果,但是触发一次PaintEvent之后,界面又恢复原样了。看了Qt源码里后发现Qt内部画的时候是相当于画在后台一个对象里,然后在刷新的时候调用bitblt统一画,调用window的api并不会影响到后面的那个对象,后台刷新的代码在:qwindowsbackingstore.cpp中的flush方法(Qt5.1.1是这个名,但在Qt5.5.1里叫flushDP)。
这个或许应该是Qt的特点,把操作系统隔离开,有好的地方,在这里又觉得不好。
Qt里截屏的方式有以下两个,功能如下:
1:QScreen的grabWindow:不会触发paintEvent消息
2:QWidget的grab:会触发paintEvent消息,相当于是调用paintEvent在一张图上画。
其它还可以调用QPixmap的两个静态方法(grabWidget跟grabWindow),但看源代码发现不推荐用这两个,应该用上面那两个替代,实现的效果应该是一样的。
基于业务的需要,很明显得采用1,实现的代码也很简单
如果需要实现滚动的话,需要调用QPixmap的scroll的方法就可以实现了。然后在PaintEvent里先把截出来的图画上去,然后再画剩下的部分就可以了。
现在理解的Qwidget的刷新机制是这样的。刷新界面一般都会调用widget的update方法。这个方法在文档里有这么一句
calling update() several times normally results in just one paintEvent() call。
跟了一下这里的源码,也发现,多次调用的活,它会在里面记录一个dirtyRegion,多次调用的话,会把区域合并在一块,在触发paint消息里统一处理。
画的时候,会根据dirtyRegion的区域只处理那部分。如果你贴图贴了整个窗口的大小,后面重绘了部分区域,很可能会画重,导致显示重叠,这点注意。
由于业务上比较复杂,截屏的区域需要去算,但原理上应该是上面提到的那样,本人思路有限,如果有更好的实现方法,也可以一块交流,在此表示感谢。 http://blog.csdn.net/hpjx1987/article/details/50634194
优化的方案是滚动滚动条的时候用截屏的方式代替界面全部刷新,优化完成后,界面在滚动时效率能提升大概一倍,背景介绍完毕。
用到最主要的是QT的截屏功能
window原生api会提供截屏滚动的功能。可以用这个ScrollWindowEx这个api。它会根据相应的参数在屏幕中进行滚动相应的区域。是不是很容易?但是结果却是不理想,因为用的是Qt,控件重写过PaintEvent的方法。调用api实时能看到效果,但是触发一次PaintEvent之后,界面又恢复原样了。看了Qt源码里后发现Qt内部画的时候是相当于画在后台一个对象里,然后在刷新的时候调用bitblt统一画,调用window的api并不会影响到后面的那个对象,后台刷新的代码在:qwindowsbackingstore.cpp中的flush方法(Qt5.1.1是这个名,但在Qt5.5.1里叫flushDP)。
这个或许应该是Qt的特点,把操作系统隔离开,有好的地方,在这里又觉得不好。
Qt里截屏的方式有以下两个,功能如下:
1:QScreen的grabWindow:不会触发paintEvent消息
2:QWidget的grab:会触发paintEvent消息,相当于是调用paintEvent在一张图上画。
其它还可以调用QPixmap的两个静态方法(grabWidget跟grabWindow),但看源代码发现不推荐用这两个,应该用上面那两个替代,实现的效果应该是一样的。
基于业务的需要,很明显得采用1,实现的代码也很简单
QScreen *pScreen = QGuiApplication::primaryScreen();
pImage = pScreen->grabWindow(pWidget->winId());
如果需要实现滚动的话,需要调用QPixmap的scroll的方法就可以实现了。然后在PaintEvent里先把截出来的图画上去,然后再画剩下的部分就可以了。
现在理解的Qwidget的刷新机制是这样的。刷新界面一般都会调用widget的update方法。这个方法在文档里有这么一句
calling update() several times normally results in just one paintEvent() call。
跟了一下这里的源码,也发现,多次调用的活,它会在里面记录一个dirtyRegion,多次调用的话,会把区域合并在一块,在触发paint消息里统一处理。
画的时候,会根据dirtyRegion的区域只处理那部分。如果你贴图贴了整个窗口的大小,后面重绘了部分区域,很可能会画重,导致显示重叠,这点注意。
由于业务上比较复杂,截屏的区域需要去算,但原理上应该是上面提到的那样,本人思路有限,如果有更好的实现方法,也可以一块交流,在此表示感谢。 http://blog.csdn.net/hpjx1987/article/details/50634194
相关文章推荐
- 用ACE的Reactor模式实现网络通讯时,ACE内部用WSAEventSelect函数把网络事件与一个事件对象关联起来,目的是为了后面用WaitForMultipleObjects函数统一处理。
- Qt刷新机制的一些总结
- 面向过程就是分析出解决问题所需要的步骤,然后用函数把这些步骤一步一步实现,使用的时候一个一个依次调用就可以了;面向对象是把构成问题事务分解成各个对象,建立对象的目的不是为了完成一个步骤,而是为了描叙某个事物在整个解决问题的步骤中的行为(转)
- [转]浅析C++中虚函数的调用及对象的内部布局(利用汇编深刻理解C++虚函数底层实现机制)
- Qt 的几个核心机制总结之 元对象系统
- 分析一个API的调用机制
- Qt信号槽机制的实现(面试的感悟,猜测每一个类保存的一个信号和槽的二维表,实际使用函数指针 元对象 还有类型安全的检查设定等等)
- 使用jQuery匹配文档中所有的li元素,返回一个jQuery对象,然后通过数组下标的方式读取jQuery集合中第1个DOM元素,此时返回的是DOM对象,然后调用DOM属性innerHTML,读取该元素 包含的文本信息
- 写一个方法进行各种属性的更新,而不要每个界面设置改变都改变一个属性。就是点击应用或确定按钮时调用这个更新属性的方法,遍历所有(控制对象属性的)界面控件的状态(值),进行属性修改和刷新。
- Window Service 创建:在后台执行一个服务,可定时做一些操作,如轮询数据库,定时发邮件
- JAVA调用系统命令或可执行程序--返回一个Runtime运行时对象,然后启动另外一个进程来执行命令
- Delphi动态事件深入分析(对象方法在调用的时候会传递一个隐含的Self指针,而该指针的值在EAX中。即左边第一个参数)
- Qt 的几个核心机制总结之 元对象系统
- 当一个对象实例作为一个参数被传递到方法中时,参数的值就是对该对象的引用。对象的内容可以在被调用的方法中改变,但对象的引用是永远不会改变的.
- [转]浅析C++中虚函数的调用及对象的内部布局(利用汇编深刻理解C++虚函数底层实现机制)
- PHP在后台启动Microsoft Word、打开一个新文件、键入一些文本、保存该文件然后关闭应用程序
- 调用WINAPI 创建一个窗口并显示一些东西(create a window and display something)
- 我这边测了一下,发现#后面参数变化浏览器不会刷新,但是#一旦去掉就会刷新了,你那边的url拼的时候能不能在没参数的时候#也拼在里面,这样应该就OK了
- 一个调用接口的时候返回不到数据分析的总结
- 在调用Qt库来实现功能过程中的一些总结