c++的陷阱(知乎)

 

原链接:https://www.zhihu.com/answer/1858690571

 

摘录的重点:

 

作者:程序喵大人
链接:https://www.zhihu.com/question/26901409/answer/1858690571
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

  • 无符号整数的错误使用()
for (unsigned int i = 10; i >= 0; --i) { ... } 

上面这段代码会发生什么? 会死循环,这里要注意下无符号整数的使用。

 

 

  • 容器的size()返回类型是无符号整数
std::vector<int> vec; vec.push_back(1); for (auto idx = vec.size(); idx >= 0; idx--) {     cout << ===== \n; } 

这段代码依旧会出现死循环,原因参考上一条。

 

 

  • memcpy、memset只适用于POD结构

至于什么是POD类型,其实解释起来挺麻烦的,感兴趣的可以直接看cppreference的

C++ named requirements: PODType​en.cppreference.com/w/cpp/named_req/PODType
  • STL遍历删除时注意迭代器失效问题
void erase(std::vector<int> &vec, int a) {     for (auto iter = vec.begin(); iter != vec.end();) { // 这个正确         if (*iter == a) {             iter = vec.erase(iter);         } else {             ++iter;         }     }      for (auto iter = vec.begin(); iter != vec.end(); ++iter) {  // error         if (*iter == a) {             vec.erase(iter); // error         }     } } 
  • std::list排序使用自己的成员方法

一般的容器排序都使用std::sort(),但是list特殊。

int main() {     std::list<int> list{1, 2, 3, 2};     list.sort();     // std::sort(list.begin(), list.end());     for (auto i : list) {         std::cout << i <<  ;     }     std::cout << \n;     return 0; } 
  • new/delete、new[]/delete[]、malloc/free严格配对

这几个一定要配对使用,原因的话可以看我之前的文章:

程序喵大人:new和delete以及new[]和delete[]一定要配对使用吗12 赞同 · 11 评论文章
  • 基类析构函数要是虚函数

如果不是虚函数的话,可能会有内存泄漏的问题

  • 注释用/**/,而不是//

注释用/**/,可能会出问题。原因:utf-8和ANSC(GB2312)编码混乱后,中文注释就乱码了,乱码中藏着 */,匹配错了,导致IDE实际注释的部分并非肉眼所见,定位极其困难,常见于Windows中。

  • 成员变量初始化

成员变量没有默认初始化行为,需要手动初始化。

  • 不要返回局部变量的指针或引用
char* func() {     char a[3] = {'a', 'b', 'c'};     return a; } 

栈内存容易被污染。

  • 浮点数判断是否相等问题
float f; if (f == 0.2) {} // 错误用法 if (abs(f - 0.2) < 0.00001) {} // 正确用法 
  • vector clear和swap问题

清空某个vector,可以使用swap而不是其clear方法,这样可以更早的释放vector内部内存。

vector<int> vec; vector<int>().swap(vec); vec.clear(); 
  • vector问题

尽量不要在vector中存放bool类型,vector为了做优化,它的内部存放的其实不是bool。