23 优先选择非成员非友元函数,而不是成员函数
概述
- 优先考虑使用非成员非友元函数,而不是成员函数
- 这种做法有助于保持类的接口简洁,提高代码的可读性和可维护性
为什么优先选择非成员非友元函数
- 增强封装性:
- 非成员函数不需要访问类的私有或受保护成员,因此可以减少类的内部实现细节暴露
- 提高代码重用性:
- 非成员函数可以作用于多个类,增强了代码的通用性和重用性
- 简化类定义:
- 将不需要访问私有成员的功能从类定义中移出,可以使类的定义更简洁、清晰
- 支持对称操作:
- 对于某些操作,如对称的二元运算符,使用非成员函数可以使代码更直观
示例代码
- 使用成员函数
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
#include <iostream> class Rectangle { private: double width; double height; public: Rectangle(double w, double h) : width(w), height(h) {} double area() const { return width * height; } double perimeter() const { return 2 * (width + height); } }; int main() { Rectangle rect(5.0, 3.0); std::cout << "Area: " << rect.area() << std::endl; std::cout << "Perimeter: " << rect.perimeter() << std::endl; return 0; } |
- 使用非成员非友元函数
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
#include <iostream> class Rectangle { private: double width; double height; public: Rectangle(double w, double h) : width(w), height(h) {} double getWidth() const { return width; } double getHeight() const { return height; } }; // 非成员非友元函数 double area(const Rectangle& rect) { return rect.getWidth() * rect.getHeight(); } double perimeter(const Rectangle& rect) { return 2 * (rect.getWidth() + rect.getHeight()); } int main() { Rectangle rect(5.0, 3.0); std::cout << "Area: " << area(rect) << std::endl; std::cout << "Perimeter: " << perimeter(rect) << std::endl; return 0; } |
24 当类型转换应该应用于所有参数时,声明非成员函数
概述
- 在设计函数时,如果希望类型转换能够应用于函数的所有参数,应将该函数声明为非成员函数
- 这是因为成员函数的第一个参数是隐式的
this
指针,而类型转换不适用于this
指针
为什么声明非成员函数
- 类型转换的应用:
- 非成员函数对所有参数都显式声明,因此编译器可以对所有参数执行类型转换
- 而对于成员函数,第一个参数是隐式的
this
指针,类型转换不适用
- 对称性和一致性:
- 对于对称操作(如算术运算符),使用非成员函数可以确保所有参数被一致地处理
- 灵活性:
- 非成员函数可以定义在类的外部,从而增加了代码的灵活性和可重用性
示例代码
- 成员函数的示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 |
#include <iostream> class Complex { private: double real; double imag; public: Complex(double r, double i) : real(r), imag(i) {} // 成员函数加法运算符 Complex operator+(const Complex& other) const { return Complex(real + other.real, imag + other.imag); } void display() const { std::cout << "(" << real << ", " << imag << ")" << std::endl; } }; int main() { Complex c1(1.0, 2.0); Complex c2(3.0, 4.0); Complex c3 = c1 + c2; // 调用成员函数 c3.display(); return 0; } |
- 非成员函数的示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 |
#include <iostream> class Complex { private: double real; double imag; public: Complex(double r, double i) : real(r), imag(i) {} // 访问私有成员的友元函数 friend Complex operator+(const Complex& lhs, const Complex& rhs); void display() const { std::cout << "(" << real << ", " << imag << ")" << std::endl; } }; // 非成员函数加法运算符 Complex operator+(const Complex& lhs, const Complex& rhs) { return Complex(lhs.real + rhs.real, lhs.imag + rhs.imag); } int main() { Complex c1(1.0, 2.0); Complex c2(3.0, 4.0); Complex c3 = c1 + c2; // 调用非成员函数 c3.display(); return 0; } |
25 考虑支持不抛出异常的swap操作
概述
- 在设计类时,考虑提供一个不抛出异常的
swap
操作 - 这对于提高代码的异常安全性和性能非常重要,特别是在实现复制并交换(
copy-and-swap
)习惯用法时
为什么需要不抛出异常的swap
- 异常安全性:
- 如果
swap
操作不抛出异常,可以在实现异常安全的赋值运算符时使用copy-and-swap
习惯用法,从而保证强异常安全性
- 如果
- 性能:
swap
操作通常是非常高效的,因为它只涉及指针或引用的交换,而不是实际数据的复制- 这使得它在性能敏感的场景中非常有用
如何实现不抛出异常的swap
- 自定义
swap
函数:- 在类中提供一个自定义的
swap
函数,并确保它不抛出异常
- 在类中提供一个自定义的
std::swap
特化:- 为
std::swap
提供特化版本,以便标准库可以使用这个高效的swap
操作
- 为
示例代码
- 自定义
swap
函数
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 |
#include <iostream> #include <utility> // for std::swap class MyClass { private: int* data; size_t size; public: // 构造函数 MyClass(size_t s) : size(s), data(new int[s]) {} // 析构函数 ~MyClass() { delete[] data; } // 复制构造函数 MyClass(const MyClass& other) : size(other.size), data(new int[other.size]) { std::copy(other.data, other.data + other.size, data); } // 赋值运算符使用copy-and-swap习惯用法 MyClass& operator=(MyClass other) { swap(other); return *this; } // 自定义swap函数 void swap(MyClass& other) noexcept { std::swap(size, other.size); std::swap(data, other.data); } // 显示数据 void display() const { for (size_t i = 0; i < size; ++i) { std::cout << data[i] << " "; } std::cout << std::endl; } }; // 为std::swap提供特化版本 namespace std { template <> void swap(MyClass& a, MyClass& b) noexcept { a.swap(b); } } int main() { MyClass obj1(5); MyClass obj2(10); std::swap(obj1, obj2); // 使用自定义的swap函数 obj1.display(); obj2.display(); return 0; } |
- 解析:
- 在类中定义一个名为
swap
的成员函数- 这个函数交换两个对象的内部状态(如指针和大小),并确保它不抛出异常(通过在函数声明中使用
noexcept
关键字)
- 这个函数交换两个对象的内部状态(如指针和大小),并确保它不抛出异常(通过在函数声明中使用
1 2 3 4 |
void swap(MyClass& other) noexcept { std::swap(size, other.size); std::swap(data, other.data); } |
- 使用
copy-and-swap
习惯用法:- 在赋值运算符中,使用
copy-and-swap
习惯用法 - 通过值传递参数并在函数体内进行交换,可以确保赋值操作的异常安全性
- 在赋值运算符中,使用
1 2 3 4 |
MyClass& operator=(MyClass other) { swap(other); return *this; } |
- 为
std::swap
提供特化版本:- 在
std
命名空间中为MyClass
提供特化的swap
版本,使得标准库在需要交换MyClass
对象时可以使用高效的自定义swap
函数
- 在
1 2 3 4 5 6 |
namespace std { template <> void swap(MyClass& a, MyClass& b) noexcept { a.swap(b); } } |
26 尽可能延迟变量定义
概述
- 在编写代码时,尽量延迟变量的定义,直到你确切知道需要使用它们为止
- 这种做法有助于提高代码的效率和可读性,减少错误的可能性,并且确保变量在尽可能小的作用域内有效
为什么要延迟变量定义
- 提高效率:
- 通过延迟变量定义,避免了不必要的变量初始化和内存分配,从而提高程序的效率
- 减少错误:
- 延迟变量定义可以减少未初始化变量的使用以及无意中覆盖变量的风险
- 增强可读性:
- 将变量的定义和使用放在一起,可以使代码更易于阅读和理解
- 缩小作用域:
- 延迟变量定义可以确保变量在尽可能小的作用域内有效,从而减少变量的生命周期,防止意外的变量覆盖和未定义行为
示例代码
- 早期定义变量(不推荐)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
#include <iostream> #include <string> void processString(bool condition) { std::string str = "Default"; // 早期定义变量 if (condition) { str = "Condition is true"; } std::cout << str << std::endl; } int main() { processString(true); processString(false); return 0; } |
- 延迟变量定义(推荐)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 |
#include <iostream> #include <string> void processString(bool condition) { if (condition) { std::string str = "Condition is true"; // 延迟变量定义 std::cout << str << std::endl; } else { std::string str = "Default"; // 延迟变量定义 std::cout << str << std::endl; } } int main() { processString(true); processString(false); return 0; } |
27 最小化类型转换
概述
- 在编写代码时尽量减少类型转换的使用
- 类型转换是一种强制改变数据类型的操作,它可能会导致代码变得复杂和容易出错
- 通过最小化类型转换,可以提高代码的可读性、可维护性和安全性
为什么要最小化类型转换
- 提高代码的可读性:
- 类型转换会使代码变得复杂和难以理解,减少类型转换可以使代码更加直观
- 减少错误:
- 类型转换可能会引入难以发现的错误,特别是在类型转换失败或导致数据损失的情况下
- 增强类型安全:
- 类型转换会绕过类型检查,增加类型不匹配的风险。通过减少类型转换,可以增强类型安全性
- 提高可维护性:
- 类型转换会增加代码的复杂性,使得代码更难以维护和修改
示例代码
- 不推荐的做法:频繁使用类型转换
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 |
#include <iostream> class Base { public: virtual ~Base() = default; }; class Derived : public Base { public: void specificFunction() { std::cout << "Derived specific function" << std::endl; } }; void process(Base* b) { // 强制类型转换 Derived* d = dynamic_cast<Derived*>(b); if (d) { d->specificFunction(); } else { std::cout << "Invalid cast" << std::endl; } } int main() { Base b; Derived d; process(&b); // 无效转换 process(&d); // 有效转换 return 0; } |
- 推荐的做法:避免类型转换
- 这个例子中,通过使用多态和虚函数,可以避免类型转换
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
#include <iostream> #include <memory> class Base { public: virtual ~Base() = default; virtual void process() = 0; // 使用多态 }; class Derived : public Base { public: void process() override { specificFunction(); } void specificFunction() { std::cout << "Derived specific function" << std::endl; } }; void processObject(std::unique_ptr<Base> b) { b->process(); // 使用多态处理 } int main() { std::unique_ptr<Base> b = std::make_unique<Derived>(); processObject(std::move(b)); // 有效处理 return 0; } |
28 避免返回指向对象内部的句柄
概述
- 免返回指向对象内部实现的句柄或指针
- 这样做的目的是为了保护对象的封装性,防止外部代码直接修改对象的内部状态,从而确保对象的一致性和完整性
为什么要避免返回指向对象内部的句柄
- 保护封装性:
- 封装是面向对象编程的基本原则之一,通过隐藏对象的内部实现细节,可以防止外部代码直接访问和修改对象的内部状态
- 确保数据一致性:
- 如果外部代码可以直接修改对象的内部数据,可能会导致数据的不一致和错误
- 通过避免返回内部句柄,可以确保所有对数据的修改都通过类的公共接口进行,从而保持数据的一致性
- 提高可维护性:
- 隐藏内部实现细节可以使类的实现更加灵活
- 内部实现可以随时修改,而不影响外部代码,从而提高代码的可维护性
- 增强安全性:
- 避免返回内部句柄可以防止外部代码进行不安全的操作,从而提高代码的安全性
示例代码
- 错误的做法:返回内部指针
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
#include <iostream> #include <string> class MyClass { private: std::string data; public: MyClass(const std::string& str) : data(str) {} // 错误:返回内部指针 std::string* getData() { return &data; } }; int main() { MyClass obj("Hello"); std::string* pData = obj.getData(); *pData = "World"; // 直接修改内部数据 std::cout << obj.getData() << std::endl; // 输出 "World" return 0; } |
- 正确的做法:返回常量引用或副本
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 |
#include <iostream> #include <string> class MyClass { private: std::string data; public: MyClass(const std::string& str) : data(str) {} // 返回常量引用 const std::string& getData() const { return data; } // 或者返回副本 std::string getDataCopy() const { return data; } }; int main() { MyClass obj("Hello"); const std::string& dataRef = obj.getData(); // dataRef = "World"; // 错误:常量引用不能修改数据 std::string dataCopy = obj.getDataCopy(); dataCopy = "World"; // 修改副本,不影响原数据 std::cout << obj.getData() << std::endl; // 输出 "Hello" return 0; } |
29 努力编写异常安全的代码
概述
- 写异常安全的代码是为了确保程序在发生异常时仍能保持一致性,并且没有资源泄漏
- 异常安全性分为三种保证级别:
- 基本保证
- 强保证
- 不抛出异常保证
基本保证
- 即使发生异常,程序状态仍然保持一致,没有资源泄漏
- 可能会有部分操作失败,但数据结构和资源保持有效
- 例如,容器在插入元素时可能会抛出异常,但容器仍然是有效的,已经存在的元素不会丢失
强保证
- 即使发生异常,程序状态仍然保持一致,没有资源泄漏,并且所有操作要么完全成功,要么完全失败(即程序状态恢复到操作前的状态)
- 这种保证通常通过使用事务性操作来实现
- 例如,插入元素时,如果操作失败,容器会恢复到插入前的状态
不抛出异常保证
- 操作承诺不会抛出任何异常
- 这种保证通常用于析构函数和
swap
操作等 - 例如,
std::vector::swap
保证不抛出异常
- 这种保证通常用于析构函数和
如何实现异常安全的代码
- 使用RAII(资源获取即初始化):
- 使用RAII管理资源,通过构造函数获取资源,通过析构函数释放资源
- 这样可以确保资源在异常发生时也能正确释放
- 例如,使用智能指针(如
std::unique_ptr
和std::shared_ptr
)来管理动态内存
- 使用RAII管理资源,通过构造函数获取资源,通过析构函数释放资源
- 使用智能指针:
- 智能指针自动管理动态内存的生命周期,确保在异常发生时内存不会泄漏
- 例如,使用
std::unique_ptr
管理动态分配的对象
- 使用标准库容器:
- 使用标准库容器(如
std::vector
、std::map
等)来管理动态数组和集合,避免手动管理内存和资源
- 使用标准库容器(如
- 使用
try
-catch
块:- 在适当的位置使用
try
-catch
块捕获异常,并进行适当的处理或清理工作 - 确保在捕获异常后程序能够恢复到一致状态
- 在适当的位置使用
- 提供异常安全的接口:
- 设计类和函数时,提供异常安全的接口,确保在发生异常时对象状态保持一致
示例代码
- 基本保证
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 |
#include <iostream> #include <vector> void process(std::vector<int>& vec, int value) { try { vec.push_back(value); } catch (const std::exception& e) { std::cerr << "Exception caught: " << e.what() << std::endl; // vec仍然是有效的,没有资源泄漏 } } int main() { std::vector<int> vec = {1, 2, 3}; process(vec, 4); for (int n : vec) { std::cout << n << " "; } std::cout << std::endl; return 0; } |
- 强保证
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
#include <iostream> #include <vector> class MyClass { public: std::vector<int> data; void addValue(int value) { std::vector<int> temp = data; // 创建临时副本 temp.push_back(value); // 对临时副本进行操作 data = temp; // 将临时副本赋值回原对象 } }; int main() { MyClass obj; obj.addValue(1); obj.addValue(2); obj.addValue(3); for (int n : obj.data) { std::cout << n << " "; } std::cout << std::endl; return 0; } |
- 不抛出异常保证
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 |
#include <iostream> #include <vector> #include <algorithm> class MyClass { public: std::vector<int> data; void swap(MyClass& other) noexcept { std::swap(data, other.data); } }; int main() { MyClass obj1; MyClass obj2; obj1.data = {1, 2, 3}; obj2.data = {4, 5, 6}; obj1.swap(obj2); for (int n : obj1.data) { std::cout << n << " "; } std::cout << std::endl; for (int n : obj2.data) { std::cout << n << " "; } std::cout << std::endl; return 0; } |
30 理解内联函数的方方面面
概述
- 内联函数(
inline functions
)是一种在函数调用时通过在调用点插入函数代码来替代函数调用,从而避免函数调用开销的方法 - 在
C++
中,内联函数可以通过在函数定义前加上inline
关键字来实现 - 内联函数具有许多优点,但也需要谨慎使用,以避免代码膨胀和其他潜在问题
为什么使用内联函数
- 减少函数调用开销:
- 内联函数通过在调用点直接插入函数代码,消除了函数调用的开销,如参数传递和返回地址保存
- 提高性能:
- 在某些情况下,内联函数可以提高性能,特别是对于短小、频繁调用的函数
- 增加编译器优化机会:
- 内联函数使得编译器有更多的机会进行优化,如常量传播和循环展开
优点
- 消除函数调用开销:
- 内联函数避免了函数调用的额外开销,特别适用于短小且频繁调用的函数
- 编译器优化:
- 内联函数提供了更多的优化机会,如常量传播、代码移动和循环展开
缺点
- 代码膨胀:
- 内联函数会增加生成代码的大小,特别是当函数在多个调用点被内联时
- 这可能导致代码膨胀,影响程序的内存占用和缓存性能
- 调试困难:
- 内联函数使得调试更加复杂,因为内联代码不会显示在栈回溯中,难以跟踪函数调用
- 编译时间增加:
- 内联函数可能增加编译时间,因为编译器需要处理更多的代码展开
使用
- 定义内联函数
1 2 3 |
inline int add(int a, int b) { return a + b; } |
- 类中的内联函数
1 2 3 4 5 6 7 8 |
class MyClass { public: inline int getValue() const { return value; } private: int value; }; |
示例代码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 |
#include <iostream> #include <chrono> // 内联函数定义 inline int add(int a, int b) { return a + b; } // 非内联函数定义 int subtract(int a, int b) { return a - b; } int main() { // 使用内联函数 auto start_inline = std::chrono::high_resolution_clock::now(); for (int i = 0; i < 100000000; ++i) { volatile int result = add(3, 4); // 使用volatile防止优化 } auto end_inline = std::chrono::high_resolution_clock::now(); // 使用非内联函数 auto start_non_inline = std::chrono::high_resolution_clock::now(); for (int i = 0; i < 100000000; ++i) { volatile int result = subtract(7, 4); // 使用volatile防止优化 } auto end_non_inline = std::chrono::high_resolution_clock::now(); // 计算并显示时间 std::chrono::duration<double> inline_duration = end_inline - start_inline; std::chrono::duration<double> non_inline_duration = end_non_inline - start_non_inline; std::cout << "Inline duration: " << inline_duration.count() << " seconds" << std::endl; std::cout << "Non-inline duration: " << non_inline_duration.count() << " seconds" << std::endl; return 0; } |
31 最小化文件之间的编译依赖
概述
- 建议尽量减少头文件之间的编译依赖关系
- 减少编译依赖可以显著加快编译速度,减少代码耦合,提高代码的可维护性和可重用性
为什么要最小化编译依赖
- 加快编译速度:
- 减少文件间的依赖关系,可以避免不必要的重新编译,提高整体编译效率
- 减少代码耦合:
- 减少依赖关系,可以降低代码模块之间的耦合,使得代码更易于维护和重用
- 提高代码的可维护性:
- 当修改一个头文件时,减少依赖关系可以减少受影响的文件数量,降低维护成本
- 避免循环依赖:
- 过多的依赖关系容易导致循环依赖,增加调试和解决问题的难度
如何最小化编译依赖
- 前向声明(
Forward Declaration
):- 在头文件中使用前向声明,而不是包含其他头文件
- 前向声明是在使用类之前声明类名,而不是包含类的完整定义
- 这可以减少头文件之间的依赖
1 |
class B; // 前向声明 |
- 使用指针或引用:
- 在类成员中使用指针或引用,而不是包含其他类的实例
- 这样可以避免在头文件中包含其他类的定义
1 |
B* b; // 使用指针 |
-
使用抽象类或接口:
- 通过抽象类或接口定义依赖关系,而不是具体实现
-
将实现细节放在源文件中:
- 尽量将类的实现细节放在源文件中,而不是头文件中
- 这样可以减少头文件的修改频率,减少编译依赖
1 |
// A.cpp 中实现 A 的构造函数、析构函数和成员函数 |
示例代码
- 错误的做法:直接包含头文件
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
// A.h #include "B.h" class A { private: B b; // 直接包含B的实例 public: void doSomething(); }; // B.h class B { public: void performAction(); }; |
- 正确的做法:使用前向声明和指针
- 在类成员中使用指针或引用,而不是包含其他类的实例
- 这样可以避免在头文件中包含其他类的定义
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 |
// A.h class B; // 前向声明 class A { private: B* b; // 使用指针 public: A(); ~A(); void doSomething(); }; // B.h class B { public: void performAction(); }; // A.cpp #include "A.h" #include "B.h" A::A() : b(new B) {} A::~A() { delete b; } void A::doSomething() { b->performAction(); } |
32 确保公共继承符合“is-a”关系
概述
- 在使用公共继承时,必须确保子类是一个真正的基类的特例,即子类和基类之间存在“
is-a
”(是一个)关系 - 公共继承表示子类不仅继承了基类的接口,还继承了基类的行为,因此子类应该能够完全替代基类的对象
为什么要确保公共继承符合“is-a”关系
Liskov
替换原则(Liskov Substitution Principle
):- 子类对象应该能够替代基类对象而不改变程序的正确性
- 违反这一原则可能导致程序的不稳定和错误
- 代码可读性和可维护性:
- 确保公共继承符合“
is-a
”关系可以使代码更易于理解和维护,避免不必要的复杂性
- 确保公共继承符合“
- 多态性:
- 在面向对象编程中,多态性的前提是子类能够替代基类对象进行操作
- 如果不符合“
is-a
”关系,多态性就不能正确实现
如何确保公共继承符合“is-a”关系
- 设计时考虑“
is-a
”关系:- 在设计类继承层次时,始终考虑子类是否真正是基类的一种特例,是否满足“
is-a
”关系
- 在设计类继承层次时,始终考虑子类是否真正是基类的一种特例,是否满足“
- 使用接口和抽象基类:
- 将公共行为抽象为接口或抽象基类,使得子类必须实现这些行为,确保接口的一致性
- 测试替换性:
- 通过编写测试代码,确保子类对象能够在所有基类对象可以使用的地方正确工作
示例代码
- 正确的公共继承示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
#include <iostream> // 基类 class Bird { public: virtual void fly() const { std::cout << "Bird is flying" << std::endl; } virtual ~Bird() = default; // 虚析构函数 }; // 子类 class Sparrow : public Bird { public: void fly() const override { std::cout << "Sparrow is flying" << std::endl; } }; void makeBirdFly(const Bird& bird) { bird.fly(); } int main() { Sparrow sparrow; makeBirdFly(sparrow); // Sparrow可以替代Bird return 0; } |
- 错误的公共继承示例
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 |
#include <iostream> // 基类 class Bird { public: virtual void fly() const { std::cout << "Bird is flying" << std::endl; } virtual ~Bird() = default; // 虚析构函数 }; // 子类 class Penguin : public Bird { public: void fly() const override { std::cout << "Penguin can't fly" << std::endl; } }; void makeBirdFly(const Bird& bird) { bird.fly(); } int main() { Penguin penguin; makeBirdFly(penguin); // Penguin不能替代Bird,违反“is-a”关系 return 0; } |
33 避免隐藏继承的名称
概述
- 在
C++
中,派生类(子类)可以继承基类的成员函数和数据成员 - 但是,如果子类定义了与基类同名的成员函数或数据成员,基类的同名成员将会被隐藏
- 隐藏继承的名称可能会导致代码的可读性和可维护性问题,以及意想不到的行为
- 因此,避免隐藏继承的名称是编写健壮和可维护代码的重要原则
为什么要避免隐藏继承的名称
- 代码可读性:
- 隐藏继承的名称会使代码变得难以理解,尤其是当同名函数在子类和基类中有不同的行为时
- 意外行为:
- 隐藏继承的名称可能会导致意外行为,特别是当开发者期望调用基类的函数而实际上调用了子类的函数时
- 维护困难:
- 在大型项目中,隐藏继承的名称会增加代码的复杂性和维护难度
示例代码
- 隐藏继承名称的问题
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
#include <iostream> class Base { public: void display() const { std::cout << "Base display" << std::endl; } }; class Derived : public Base { public: // 隐藏了基类的display函数 void display() const { std::cout << "Derived display" << std::endl; } }; int main() { Derived d; d.display(); // 调用的是Derived的display d.Base::display(); // 调用基类的display(需要显式调用) return 0; } |
避免隐藏继承名称的方法
- 使用
using
声明- 在子类中使用
using
声明来引入基类的同名成员,从而避免隐藏
- 在子类中使用
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
#include <iostream> class Base { public: void display() const { std::cout << "Base display" << std::endl; } }; class Derived : public Base { public: // 使用using声明引入基类的display函数 using Base::display; void display(int i) const { std::cout << "Derived display with int: " << i << std::endl; } }; int main() { Derived d; d.display(); // 调用Base的display d.display(10); // 调用Derived的display(int) return 0; } |
- 选择不同的名称
- 为子类中的新函数选择不同的名称,以避免与基类中的函数同名
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 |
#include <iostream> class Base { public: void display() const { std::cout << "Base display" << std::endl; } }; class Derived : public Base { public: void show() const { std::cout << "Derived show" << std::endl; } }; int main() { Derived d; d.display(); // 调用Base的display d.show(); // 调用Derived的show return 0; } |
本文为原创文章,版权归Aet所有,欢迎分享本文,转载请保留出处!
你可能也喜欢
- ♥ C++标准模板库编程实战_适配器12/07
- ♥ 51CTO:C++语言高级课程二08/08
- ♥ Boost 程序库完全开发指南:容器算法数学文件08/24
- ♥ 线程和协程10/31
- ♥ 深入理解C++11:C++11新特性解析与应用 三01/05
- ♥ C++_函数模板、类模板、特化、模板元编程、SFINAE、概念06/22