0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

C++语言里有哪些比较常见的坑?

Q4MP_gh_c472c21 来源:程序喵大人 作者:程序喵大人 2021-05-20 11:38 次阅读

前段时间给部门做了个C++专题的分享,主要分享了C++语言里一些常见的坑,在这里也分享给大家。

首先说下C++和C语言有什么区别?分享一个我在知乎上看见的回答:

C++ ≈ C with classes, C with STL

C:面向机器编程

C++:面向编译器编程

C++有个很重要的特性叫RAII,个人认为可以多多使用,相当方便,关于RAII巧妙使用可以看我这两篇文章《RAII妙用之ScopeExit》《RAII妙用之计算函数耗时》。

言归正传,下面我一个一个的列出来C++使用过程中常见的坑:

无符号整数的错误使用

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 《《 “=====

”;}

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

memcpy、memset只适用于POD结构

至于什么是POD类型,其实解释起来挺麻烦的,感兴趣的可以直接看cppreference的https://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 《《 “

”; return 0;}

new/delete、new[]/delete[]、malloc/free严格配对

这几个一定要配对使用,原因的话可以看我之前的文章《new[]和delete[]为何要配对使用?》

基类析构函数要是虚函数

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

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

注释用/**/,可能会出问题。原因: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。

条件变量

条件变量的使用有两大问题:信号丢失和虚假唤醒,相当重要,具体可以看我这篇文章《使用条件变量的坑你知道吗》。

类型转换

在C++中尽量使用C++风格的四种类型转换,而不要使用C语言风格的强制类型转换。

异步操作中async的使用

std::async(std::async, []{ f(); }); // 临时量的析构函数等待 f()std::async(std::async, []{ g(); }); // f() 完成前不开始

std::async 这货返回的 future 和通过 promise 获取的 future 行为不同,async 返回的 future 对象在析构时会阻塞等待 async 中的线程执行完毕,这就导致在大部分场景中 async达不到你直觉的认为它能达到的目的。

智能指针

一个裸指针不要使用多个智能指针包裹,尽可能使用make_unique,make_shared。

当需要在类得内部接口中,需要将this作为智能指针使用,需要用该类派生自enable_shared_from_this

栈内存使用

合理使用栈内存,特别是数组,数组越界问题容易导致栈空间损坏,可以考虑使用std::array替代普通的数组。

std::thread的使用

一定要记得join或这detach,否则会crash。

void func() {}int main() { std::thread t(func); if (t.joinable()) { t.join(); // 或者t.detach(); } return 0;}

enum使用

尽量使用enum class替代enum,enum class 是带有作用域的枚举类型。

空指针使用nullptr而不是NULL

至于为什么要这么使用,可以看我这篇文章《关于nullptr这篇文章你一定要看》

void func(char*) { cout 《《 “char*”;}void func(int) { cout 《《 “int”;}

int main() { func(NULL); // 编译失败 error: call of overloaded ‘func(NULL)’ is ambiguous func(nullptr); // char* return 0;}

std::remove的使用

这个remove其实并没有真正的删除元素,需要和erase配合使用,跑一下这段代码就知道啦。

bool isOdd(int i) { return i & 1; }

void print(const std::vector《int》& vec) { for (const auto& i : vec) { std::cout 《《 i 《《 ‘ ’; } std::cout 《《 std::endl;}

int main() { std::vector《int》 v = {0, 1, 2, 3, 4, 5, 6, 7, 8, 9}; print(v);

std::remove(v.begin(), v.end(), 5); // error print(v);

v.erase(std::remove(v.begin(), v.end(), 5), v.end()); print(v);

v.erase(std::remove_if(v.begin(), v.end(), isOdd), v.end()); print(v);}

全局变量初始化问题

不同文件中的全局变量初始化顺序不固定,全局变量尽量不要互相依赖,否则由于初始化顺序不固定的问题,可能会导致bug产生。

原文标题:C++的24个大坑,你能躲过几个?

文章出处:【微信公众号:嵌入式ARM】欢迎添加关注!文章转载请注明出处。

责任编辑:haq

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 编程
    +关注

    关注

    88

    文章

    3615

    浏览量

    93710
  • C++
    C++
    +关注

    关注

    22

    文章

    2108

    浏览量

    73636

原文标题:C++的24个大坑,你能躲过几个?

文章出处:【微信号:gh_c472c2199c88,微信公众号:嵌入式微处理器】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    同样是函数,在CC++中有什么区别

    ,即使没有数据返回,也得写 void。 第二个函数名。 C语言的函数名绝对不能重名,除了用上 weak 这样的黑科技。同一个项目中,函数重名就会提示重复定义。 C++因为函数重载的存在,函数名可以相同,只要参数
    的头像 发表于 11-29 10:25 291次阅读

    NPU支持的编程语言哪些

    与NPU一起使用: C/C++CC++是性能要求较高的应用的首选语言,尤其是在需要直接与硬件交互的场景中。许多NPU硬件都提供了
    的头像 发表于 11-15 09:21 634次阅读

    C语言C++中结构体的区别

    同样是结构体,看看在C语言C++中有什么区别?
    的头像 发表于 10-30 15:11 204次阅读

    C语言与其他编程语言比较

    C语言作为一种历史悠久的编程语言,自其诞生以来,一直在软件开发领域扮演着重要角色。它以其高效、灵活和可移植性强的特点,成为了系统级编程的首选语言之一。
    的头像 发表于 10-29 17:30 270次阅读

    嵌入式QT常见开发方式哪些?

    嵌入式QT常见开发方式哪些? 嵌入式工程师在学习和使用Qt进行开发时,常见的几种开发方式包括: 1.Qt Widgets编程: 通过C++代码直接编写GUI应用程序,利用QtWi
    发表于 08-12 10:05

    C++语言基础知识

    电子发烧友网站提供《C++语言基础知识.pdf》资料免费下载
    发表于 07-19 10:58 7次下载

    C++中实现类似instanceof的方法

    C++多态与继承,但是很多人开始学习C++,有时候会面临一个常见问题,就是如何向下转型,特别是不知道具体类型的时候,这个时候就希望C++
    的头像 发表于 07-18 10:16 582次阅读
    <b class='flag-5'>C++</b>中实现类似instanceof的方法

    SEGGER编译器优化和安全技术介绍 支持最新CC++语言

    SEGGER编译器是专门为ARM和RISC-V微控制器设计的优化C/C++编译器。它建立在强大的Clang前端上,支持最新的CC++语言
    的头像 发表于 06-04 15:31 1456次阅读
    SEGGER编译器优化和安全技术介绍 支持最新<b class='flag-5'>C</b>和<b class='flag-5'>C++</b><b class='flag-5'>语言</b>

    C/C++中两种宏实现方式

    #ifndef的方式受C/C++语言标准支持。它不仅可以保证同一个文件不会被包含多次,也能保证内容完全相同的两个文件(或者代码片段)不会被不小心同时包含。
    的头像 发表于 04-19 11:50 609次阅读

    为什么很少用C++开发单片机

    C语言是面向过程的语言C++是面向对象的编程语言。结合本文来说,面向过程相比面向对象的编程,生成代码量(bin文件)更小,运行效率更高。
    发表于 03-25 14:26 961次阅读
    为什么很少用<b class='flag-5'>C++</b>开发单片机

    C语言#define的应用

    C/C++ 编程语言中,当程序被编译时,被发送到编译器,编译器将程序转换为机器语言,然后完成编译并执行该程序。预处理器也称为宏预处理器。
    发表于 03-06 11:29 376次阅读
    <b class='flag-5'>C</b><b class='flag-5'>语言</b>#define的应用

    plc编程语言c语言的联系 c语言和PLC什么区别

    语言,主要用于开发各种应用程序。尽管PLC编程语言C语言一些相似之处,但它们之间也存在一些明显的区别。 首先,PLC编程
    的头像 发表于 02-05 14:21 4110次阅读

    c语言,c++,java,python区别

    C语言C++、Java和Python是四种常见的编程语言,各有优点和特点。 C
    的头像 发表于 02-05 14:11 2384次阅读

    vb语言c++语言的区别

    VB语言C++语言是两种不同的编程语言,虽然它们都属于高级编程语言,但在设计和用途上有很多区别。下面将详细
    的头像 发表于 02-01 10:20 2291次阅读

    C++简史:C++是如何开始的

    的 MISRA C++:2023 博客系列的第二部分。 在这篇博客中,我们将深入探讨 C++ 的历史、编程语言多年来的发展历程以及它的下一步发展方向。
    的头像 发表于 01-11 09:00 588次阅读
    <b class='flag-5'>C++</b>简史:<b class='flag-5'>C++</b>是如何开始的