时间:2021-05-20
静态代码分析工具可简化编码过程,检测出错误并帮助修复。PVS-Studio 是一个用于 C/C++ 的静态代码分析工具。该团队检测了 200 多个C/C++开源项目,包括了 Unreal Engine、Php、Haiku、Qt 和 Linux 内核等知名项目。于是他们每天分享一个错误案例,并给出相应建议。
这个bug是在UnrealEngine4 的源代码中发现的。
错误代码:
void FSlateNotificationManager::GetWindows( TArray< TSharedRef<SWindow> >& OutWindows) const{ for( auto Iter(NotificationLists.CreateConstIterator()); Iter; Iter++ ) { TSharedPtr<SNotificationList> NotificationList = *Iter; .... }}解释:
如果不读标题的话,你可能很难发现这段代码里的问题。第一眼看上去这段代码完全正确,其实它并不完美。没错,我指的是后自增运算符 Iter++ 。我们应该尽量使用前自增运算符而不是后自增运算符,即用 ++Iter 代替 Iter++ 。为什么要这么做,有什么有实际价值?下面我会详细解释。
正确代码:
void FSlateNotificationManager::GetWindows( TArray< TSharedRef<SWindow> >& OutWindows) const{ for( auto Iter(NotificationLists.CreateConstIterator()); Iter; ++Iter) { TSharedPtr<SNotificationList> NotificationList = *Iter; .... }}建议:
前缀和后缀形式之间的区别是众所周知的。我希望它们内部结构的区别(告诉了我们运算法则)大家也是清楚的。如果你有使用过运算符重载的话,肯定已经意识到了。没有用过的话,我在这儿简单地解释一下(用过运算符重载的可以跳过下面关于运算符重载的例子)。
前自增运算符改变了对象的状态并返回对象改变后的状态,不需要创建临时对象。下面是前自增运算符的例子:
MyOwnClass& operator++(){ ++meOwnField; return (*this);}后自增运算符也改变了对象的状态但是返回的是对象改变前的状态,并且需要创建一个临时对象。下面是后自增运算符重载的例子:
MyOwnClass operator++(int){ MyOWnCLass tmp = *this; ++(*this); return tmp;}看到上面这段代码,你会发现有一个额外的操作,就是要创建一个临时对象,在实践中这点太重要了!
现在的编译器做代码优化的时候非常智能,如果没有用处,是不会随便创建临时对象的。这就是为什么在发布版中我们很难发现i++和++i的区别。
但是在调试模式下进行程序调试的时候就是另一回事了,这时候你会看到性能上有很大差别。
有一些例子可以估计调试版本中使用前自增和后自增运算符的代码运行时间,我们可以看到使用后缀形式所用时间几乎是前缀的四倍。
有人会说:”那又怎么样?反正发布版都是一样的。”,这种想法说对也对说不对也不对。通常我们会花更多的时间做单元测试和调试程序,所以大多数时间都在调试版本下工作,谁也不想浪费时间在那儿等吧?
关于“对于迭代器,我们是否应该用前自增运算符(++i)来代替后自增运算符(i++)?”这个问题,我想认真地回答:“是的,真应该这么做”。你会发现在调试版本中速度大大提升。如果迭代器很复杂的话,这么做的好处更是显而易见了。
这个错误是用静态代码分析工具PVS-Studio发现的,错误信息为:V803性能下降。如果iter是迭代器的话,使用前自增运算符会更高效,使用++iter代替iter++.
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对的支持。如果你想了解更多相关内容请查看下面相关链接
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
1.iterator叠代器最简单例子应该是数组下标了,且看下面的c++代码:复制代码代码如下:intarray[10];for(inti=0;i<10;i++)
i=i++;j=i++的区别i=i++-----------在java中这个语句的前后顺序应该是这样的(tmp=i;i++;tmp==i)java的编译器在遇到
i++和++i的区别: 1、i++返回原来的值,++i返回加1后的值。 2、i++不能作为左值,而++i可以。 3、i++前者是先赋值,然后再自增;++i
下面从汇编语言看其原理.下面是c++源代码:复制代码代码如下:voidadd(){staticinti=1;i++;}intmain(){add();}下面是m
breakfunctionmyBreak(){for(vari=0;i<5;i++){if(i==3){break;}console.log(i);}}myBr