C++容易犯的九个陷阱(代码版本)

2024-02-06 20:28

本文主要是介绍C++容易犯的九个陷阱(代码版本),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

自己也算入行一年了,遇到不少坑,结合网上博客以及自己复现,总结有以下几个容易出错的点

目录

1、 unsigned不够理解

2、容器的size()返回的也是无符号整型

3、 容器删除元素迭代器失效

4、注释尽量用//而不是/**/

5、成员变量没有初始化行为需要手动初始化

6、浮点数判断是否相等

7、thread使用

8、注意操作符短路

9、对象切割


1、 unsigned不够理解

代码死循环

2、容器的size()返回的也是无符号整型

代码死循环

3、 容器删除元素迭代器失效

当从 std::vector 中移除一个元素时,所有该位置之后的元素都会向前移动,这意味着迭代器、引用和指向被移动元素的指针都会失效。

#include <iostream>
#include <vector>
using namespace std;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) {  // errorif (*iter == a) {vec.erase(iter); // 错误}}
}int main() {std::vector<int> vec;vec.push_back(1);vec.push_back(2);vec.push_back(3);vec.push_back(4);erase(vec, 2);erase(vec, 3);for (auto it = vec.begin(); it != vec.end(); it++) {cout << "cur:" << *it << endl;}return 0;
}

 4、注释尽量用//而不是/**/

关于中文乱码问题

5、成员变量没有初始化行为需要手动初始化

这样做的原因是为了确保数据的一致性和预期行为,避免出现未初始化的变量导致的不确定性和潜在的错误。

class MyClass {
public:MyClass() : myInt(0), myDouble(0.0) {} // 构造函数中初始化private:int myInt;    // 这些成员变量不会自动初始化double myDouble;
};

6、浮点数判断是否相等

#include <iostream>
#include <vector>
using namespace std;int main() {float f=0.2;if (f == 0.2) {cout << "1true";} // 错误用法if (abs(f - 0.2) < 0.00001) {cout << "2true";} // 正确用法return 0;
}

7、thread使用

记得要join


#include <iostream>
#include <vector>
#include <thread>
using namespace std;void func() {}
int main() {std::thread t(func);if (t.joinable()) {t.join(); // 或者t.detach(); }return 0;
}

8、注意操作符短路

||左边只要是true 那么右边的语句不执行了

&&左边只要是false 那么右边的语句就不执行了

bool HP() {std::cout << "我被执行HP方法" << std::endl;return true;
}
bool MP() {std::cout << "我被执行MP方法" << std::endl;return false;
}
int main() {if (HP() | MP()) {std::cout << "更新客户端" << std::endl;}return 0;}

 9、对象切割

当一个派生类对象被赋值给一个基类对象时,超出基类部分的成员变量会被“切割”掉,这会导致信息丢失。这个问题通常发生在使用值传递而非引用或指针传递派生类对象时。为了避免对象切割,应当使用指针或引用来处理多态。

 错误案例

#include <iostream>
using namespace std;class Base {
public:Base() { cout << "Base constructor\n"; }virtual void print() { cout << "Base class\n"; }virtual ~Base() { cout << "Base destructor\n"; }
};class Derived : public Base {
public:Derived() { cout << "Derived constructor\n"; }void print() override { cout << "Derived class\n"; }~Derived() override { cout << "Derived destructor\n"; }void specificFunction() { cout << "Specific function for Derived\n"; }
};void functionByValue(Base b) {cout << "************关注下面这一句****************" << endl;b.print(); // 这里会调用 Base 的 print,而不是 Derived 的cout << "****************************" << endl;
}int main() {Derived d;functionByValue(d); // 将 Derived 对象以值传递给函数return 0;
}

main 函数中,我们创建了一个 Derived 类的对象 d。当我们调用 functionByValue 并将 d 传递进去时,由于函数接受的是 Base 类型的值,因此 d 对象会被切割为 Base 类型。这意味着,函数内部的 b 对象只包含 Base 类的部分,Derived 类特有的成员和行为(如 specificFunction 方法和 Derived 类的 print 方法实现)都被丢弃了。因此,即使我们传递了一个 Derived 类型的对象,函数内部调用的 print 方法仍然是 Base 类的版本。 

正确案例:

修改为引用类型

void functionByReference(Base& b) {b.print(); // 这会调用正确的 print 方法,即 Derived 的 print
}int main() {Derived d;functionByReference(d); // 通过引用传递return 0;
}

 在这个修正后的版本中,我们通过引用传递 Derived 对象,因此 functionByReference 函数内部可以正确地调用 Derived 类的 print 方法,而不会发生对象切割。

 

总结了24个C++的大坑,看你能躲过几个?-CSDN博客

这篇关于C++容易犯的九个陷阱(代码版本)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/685438

相关文章

从入门到精通C++11 <chrono> 库特性

《从入门到精通C++11<chrono>库特性》chrono库是C++11中一个非常强大和实用的库,它为时间处理提供了丰富的功能和类型安全的接口,通过本文的介绍,我们了解了chrono库的基本概念... 目录一、引言1.1 为什么需要<chrono>库1.2<chrono>库的基本概念二、时间段(Durat

C++20管道运算符的实现示例

《C++20管道运算符的实现示例》本文简要介绍C++20管道运算符的使用与实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录标准库的管道运算符使用自己实现类似的管道运算符我们不打算介绍太多,因为它实际属于c++20最为重要的

Java中调用数据库存储过程的示例代码

《Java中调用数据库存储过程的示例代码》本文介绍Java通过JDBC调用数据库存储过程的方法,涵盖参数类型、执行步骤及数据库差异,需注意异常处理与资源管理,以优化性能并实现复杂业务逻辑,感兴趣的朋友... 目录一、存储过程概述二、Java调用存储过程的基本javascript步骤三、Java调用存储过程示

Visual Studio 2022 编译C++20代码的图文步骤

《VisualStudio2022编译C++20代码的图文步骤》在VisualStudio中启用C++20import功能,需设置语言标准为ISOC++20,开启扫描源查找模块依赖及实验性标... 默认创建Visual Studio桌面控制台项目代码包含C++20的import方法。右键项目的属性:

c++中的set容器介绍及操作大全

《c++中的set容器介绍及操作大全》:本文主要介绍c++中的set容器介绍及操作大全,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录​​一、核心特性​​️ ​​二、基本操作​​​​1. 初始化与赋值​​​​2. 增删查操作​​​​3. 遍历方

解析C++11 static_assert及与Boost库的关联从入门到精通

《解析C++11static_assert及与Boost库的关联从入门到精通》static_assert是C++中强大的编译时验证工具,它能够在编译阶段拦截不符合预期的类型或值,增强代码的健壮性,通... 目录一、背景知识:传统断言方法的局限性1.1 assert宏1.2 #error指令1.3 第三方解决

C++11委托构造函数和继承构造函数的实现

《C++11委托构造函数和继承构造函数的实现》C++引入了委托构造函数和继承构造函数这两个重要的特性,本文主要介绍了C++11委托构造函数和继承构造函数的实现,具有一定的参考价值,感兴趣的可以了解一下... 目录引言一、委托构造函数1.1 委托构造函数的定义与作用1.2 委托构造函数的语法1.3 委托构造函

MySQL数据库的内嵌函数和联合查询实例代码

《MySQL数据库的内嵌函数和联合查询实例代码》联合查询是一种将多个查询结果组合在一起的方法,通常使用UNION、UNIONALL、INTERSECT和EXCEPT关键字,下面:本文主要介绍MyS... 目录一.数据库的内嵌函数1.1聚合函数COUNT([DISTINCT] expr)SUM([DISTIN

C++11作用域枚举(Scoped Enums)的实现示例

《C++11作用域枚举(ScopedEnums)的实现示例》枚举类型是一种非常实用的工具,C++11标准引入了作用域枚举,也称为强类型枚举,本文主要介绍了C++11作用域枚举(ScopedEnums... 目录一、引言二、传统枚举类型的局限性2.1 命名空间污染2.2 整型提升问题2.3 类型转换问题三、C

C++链表的虚拟头节点实现细节及注意事项

《C++链表的虚拟头节点实现细节及注意事项》虚拟头节点是链表操作中极为实用的设计技巧,它通过在链表真实头部前添加一个特殊节点,有效简化边界条件处理,:本文主要介绍C++链表的虚拟头节点实现细节及注... 目录C++链表虚拟头节点(Dummy Head)一、虚拟头节点的本质与核心作用1. 定义2. 核心价值二