C++ 面向对象知识汇总(超详细)

学习交流:0voice · GitHub

1.什么是类?

在C++中,类(Class) 是一种用户定义的数据类型,用来描述具有相同特征和行为的一组对象。类是面向对象编程(OOP)的核心概念,它通过将数据和操作封装在一起,提供了创建复杂程序的工具。

类的基本组成部分:

  1. 成员变量(Member Variables):也称为属性或数据成员,是类中用来存储对象状态的数据。
  2. 成员函数(Member Functions):也称为方法,是类中定义的用于操作成员变量或执行操作的函数。

类的声明和定义:

类通常由类声明和类定义两部分组成。

1. 类声明(Class Declaration)

这部分通常位于头文件中,定义了类的接口,包括成员变量和成员函数的声明。语法如下:

class ClassName {
public:// 公有成员函数void function1();private:// 私有成员变量int variable1;
};
  • public:表示该部分的成员可以被类外部直接访问。
  • private:表示该部分的成员只能被类内部的函数访问,外部无法直接访问。
2. 类定义(Class Definition)

类的成员函数通常在类声明之外定义(可以放在.cpp文件中),如下:

#include <iostream>class MyClass {
public:void setValue(int val) {variable = val;}int getValue() {return variable;}private:int variable;
};int main() {MyClass obj;  // 创建对象obj.setValue(10);  // 设置对象的值std::cout << "Value: " << obj.getValue() << std::endl;  // 获取对象的值return 0;
}

类的关键特性:

  1. 封装(Encapsulation):将数据和操作封装在类中,通过访问控制机制(如publicprivate)控制外部访问。

  2. 继承(Inheritance):允许一个类从另一个类派生,继承其成员和方法,支持代码复用和扩展。

  3. 多态性(Polymorphism):通过继承和虚函数实现不同对象对相同消息作出不同响应。

2.面向对象的程序设计思想是什么?

1. 封装(Encapsulation)

封装是将对象的 数据方法 封装在一个类中,隐藏其内部实现细节,只通过公开的接口(成员函数)来访问和修改数据。封装的目的是提高安全性和可维护性。

  • 优势
    • 控制对数据的访问:通过将成员变量设为 private,确保数据只能通过特定的方法(如 gettersetter 函数)来访问和修改。
    • 防止外部干扰:防止外部代码直接修改对象的内部状态,保护对象不受意外的或不合理的修改。
class Student {
private:int age;  // 私有数据,不能直接访问public:void setAge(int a) {if (a > 0) {age = a;  // 设置合法的值}}int getAge() {return age;  // 提供安全的访问方式}
};

在这个例子中,age 是一个私有属性,不能直接在类外部访问,而必须通过公开的 setAgegetAge 方法来操作。

2. 继承(Inheritance)

继承允许一个类(子类或派生类)从另一个类(父类或基类)继承属性和方法,子类可以复用父类的代码,或者根据需要对继承的功能进行扩展或修改。

  • 优势
    • 代码复用:子类继承父类的属性和方法,避免重复代码。
    • 扩展现有类:通过继承,可以在不修改父类代码的情况下,扩展或定制新的功能。
class Animal {
public:void eat() {std::cout << "Eating...\n";}
};class Dog : public Animal {  // Dog 类继承自 Animal 类
public:void bark() {std::cout << "Barking...\n";}
};

在这个例子中,Dog 类继承了 Animal 类的 eat 方法,并新增了 bark 方法。Dog 对象可以调用 eat,因为它从 Animal 类继承了这一功能。

3. 多态(Polymorphism)

多态性允许不同的对象对同一消息作出不同的响应。它通过 函数重载虚函数 实现,主要表现为 编译时多态性运行时多态性

  • 编译时多态性(静态多态性):通过 函数重载运算符重载 实现。
  • 运行时多态性(动态多态性):通过 虚函数继承 实现,基类指针或引用可以指向派生类对象,并根据实际对象类型调用相应的派生类方法。
class Animal {
public:virtual void sound() {  // 虚函数std::cout << "Animal makes a sound\n";}
};class Dog : public Animal {
public:void sound() override {  // 重写基类的虚函数std::cout << "Dog barks\n";}
};class Cat : public Animal {
public:void sound() override {std::cout << "Cat meows\n";}
};int main() {Animal* animal1 = new Dog();Animal* animal2 = new Cat();animal1->sound();  // 输出:Dog barksanimal2->sound();  // 输出:Cat meowsdelete animal1;delete animal2;return 0;
}

在这个例子中,虽然 animal1animal2 都是 Animal 类型的指针,但在调用 sound() 方法时,会根据实际对象类型(DogCat)执行相应的重写方法,这就是多态性的体现。

小结:

  1. 封装:将数据和方法封装在类中,保护数据不被外部直接修改。
  2. 继承:允许类从已有类继承属性和方法,促进代码复用和扩展。
  3. 多态:不同对象可以通过同一个接口(如虚函数)表现出不同的行为。

3.C++中struct和class有什么区别?

在C++中,structclass 都可以用于定义包含成员变量和成员函数的数据类型,它们的功能基本上是相同的。不过,二者的主要区别在于 成员的默认访问权限使用习惯

1. 默认的访问控制权限(Access Control)

struct

struct 中,成员默认是公有的(public)。这意味着,如果你不显式指定成员的访问权限,它们将自动具有公有访问权限。

struct MyStruct {int x;  // 默认是 public
};int main() {MyStruct s;s.x = 10;  // 可以直接访问,因为是 publicreturn 0;
}
class

class 中,成员默认是私有的(private)。这意味着,如果不显式指定成员的访问权限,它们默认是私有的,外部无法直接访问。

class MyClass {int x;  // 默认是 private
};int main() {MyClass c;// c.x = 10;  // 错误,不能直接访问 private 成员return 0;
}

2. 传统使用习惯

  • struct 通常用于表示 简单的数据结构,通常只包含成员变量,没有复杂的成员函数。它的用法更类似于C语言中的 struct,被视为一个数据包(data bundle)。

  • class 则更常用于定义 复杂的对象,包含数据成员和操作方法。它的使用更符合面向对象编程的思想,如封装、继承和多态。

3. 继承的访问权限

在继承时,structclass 也有一些细微差别。

struct 继承

struct 中继承的默认访问权限是 public。这意味着,如果你不指定访问控制,派生类会公开继承基类的所有成员。

struct Base {int x;
};struct Derived : Base {// 继承 Base 中的 x,默认是 public 继承
};int main() {Derived d;d.x = 10;  // 访问 x 是合法的,因为是 public 继承return 0;
}
class 继承

class 中继承的默认访问权限是 private。如果不指定继承方式,基类的成员在派生类中默认是私有的,无法在派生类对象中访问。

class Base {int x;
};class Derived : Base {// 默认是 private 继承
};int main() {Derived d;// d.x = 10;  // 错误,x 在 Derived 中是 privatereturn 0;
}

4. 常见误区

  • 有时人们认为 struct 只能包含数据成员,class 才能包含成员函数,这实际上是不对的。在C++中,structclass 都可以包含成员函数,并且支持所有的面向对象特性(如继承和多态)。

  • 使用 structclass 的选择主要基于编程习惯struct 通常用来表示简单的、仅包含数据的结构,而 class 用来表示更复杂的、带有行为的对象。

5. 小结:

特性structclass
默认成员访问权限publicprivate
默认继承访问权限publicprivate
用途习惯简单数据结构面向对象编程
支持面向对象特性

总结来说,structclass 的功能几乎是相同的,主要区别在于默认的访问控制权限。可以根据需要选择使用 structclass,但 class 更常用于复杂对象,struct 更适合用于简单数据结构。

4.动态多态有什么作用?有哪些必要条件?

动态多态是面向对象编程(OOP)中非常重要的概念,它允许对象在运行时根据它们的实际类型来选择合适的函数进行调用。这种能力使得程序更灵活、更易扩展,可以处理不同对象的多种行为,而无需在编译时确定具体的对象类型。

动态多态的作用:

  1. 提高代码灵活性和可扩展性: 动态多态允许程序对不同的对象使用相同的接口。这样,在不修改现有代码的情况下,可以为新类型的对象定义新行为,而不用改动调用这些接口的代码。

  2. 实现通用接口: 使用基类定义通用接口,派生类可以根据自己的特性实现这些接口。调用者无需知道对象的具体类型,只需要依赖基类接口,这使得代码易于维护和扩展。

  3. 减少重复代码: 动态多态通过继承和虚函数机制,让派生类能够复用基类的公共代码,同时在需要时定义自己的特殊行为,减少了代码重复。

动态多态的必要条件:

在C++中,要实现动态多态,必须满足以下三个条件:

  1. 继承(Inheritance): 动态多态依赖于类的继承机制。通常,基类提供一个接口,派生类继承基类并实现或重写基类中的方法。

  2. 虚函数(Virtual Functions): 基类中的函数必须声明为虚函数virtual),以便在运行时通过基类指针或引用调用派生类的函数。虚函数允许C++的运行时多态性,即程序在运行时根据对象的实际类型来决定调用哪个函数。

    class Base {
    public:virtual void show() {std::cout << "Base class" << std::endl;}
    };class Derived : public Base {
    public:void show() override {  // 重写基类的虚函数std::cout << "Derived class" << std::endl;}
    };
    
  3. 通过基类指针或引用访问对象: 动态多态的核心是在运行时通过基类指针或引用来调用派生类的函数。C++的虚函数表(vtable)机制确保在运行时找到正确的函数实现。

    int main() {Base* bPtr;Derived d;bPtr = &d;  // 基类指针指向派生类对象bPtr->show();  // 调用的是 Derived 类的 show() 函数,输出 "Derived class"return 0;
    }
    

动态多态的实现流程:

  • 当通过基类指针或引用调用虚函数时,C++ 会在运行时通过虚函数表找到对象的实际类型,并调用该类型对应的函数实现。
  • 如果派生类重写了基类的虚函数,则调用派生类的实现;如果没有重写,则调用基类的实现。

动态多态与静态多态的区别:

  • 动态多态:是在运行时根据对象的类型决定调用哪个函数(通过虚函数实现)。动态多态使用继承和虚函数,支持对象的行为多样性。
  • 静态多态:是在编译时确定函数调用,通常通过函数重载或模板实现。它不依赖继承和虚函数,而是在编译时进行解析。

总结:

动态多态通过虚函数和继承机制,让程序在运行时能够根据对象的实际类型选择合适的函数执行,从而提高代码的灵活性和扩展性。要实现动态多态,需要满足以下三个条件:

  1. 类的继承。
  2. 基类的函数必须是虚函数。
  3. 通过基类指针或引用访问派生类对象。

5.构造函数为什么不能是虚函数

在C++中,基类的构造函数不能被定义为虚函数,原因有两个:

  1. 构造函数的目的是初始化对象。当我们创建一个对象时,构造函数被调用来初始化对象的数据成员。在这个阶段,对象才刚刚开始被构建,还没有完全形成,因此它还不具备执行虚函数调用的条件(即,动态绑定)。因为执行虚函数调用需要通过对象的虚函数表指针,而这个指针在构造函数执行完毕后才会被设置。

  2. 虚函数通常在有继承关系的类中使用,用于实现多态。在子类对象的构造过程中,首先会调用基类的构造函数,然后才是子类的构造函数。如果基类的构造函数被定义为虚函数,那么在执行基类的构造函数时,由于子类的部分还没有被构造,所以无法正确地执行子类构造函数中对虚函数的重写。这就破坏了虚函数的目的,即允许子类重写基类的行为。

因此,基于以上原因,C++不允许构造函数为虚函数。但是,析构函数可以(并且通常应该)被声明为虚函数,以确保当删除一个指向派生类对象的基类指针时,派生类的析构函数能被正确调用,避免资源泄露。

6.为什么基类的析构函数需要定义为虚函数?

在C++中,基类的析构函数通常应该定义为虚函数,尤其是在使用继承多态时。这样做的主要原因是为了确保正确调用派生类的析构函数,避免资源泄漏和未定义行为。

问题背景

当我们通过基类指针基类引用指向一个派生类对象时,如果析构函数不是虚函数,销毁对象时可能会发生问题。举个例子:

class Base {
public:~Base() {std::cout << "Base destructor called" << std::endl;}
};class Derived : public Base {
public:~Derived() {std::cout << "Derived destructor called" << std::endl;}
};int main() {Base* obj = new Derived();  // 基类指针指向派生类对象delete obj;  // 销毁对象return 0;
}

输出

Base destructor called

在这个例子中,delete obj 时,只调用了基类 Base 的析构函数,并没有调用 Derived 类的析构函数。这就导致了派生类的资源没有被正确释放,从而可能引发内存泄漏或其他资源管理问题。因为 delete 只会调用基类的析构函数,派生类部分的对象并没有被正确销毁。

虚析构函数的必要性

要解决上述问题,需要将基类的析构函数声明为虚函数virtual),这样在通过基类指针或引用销毁派生类对象时,C++ 的虚函数机制会确保派生类的析构函数也能被正确调用。

虚析构函数的实现:
class Base {
public:virtual ~Base() {std::cout << "Base destructor called" << std::endl;}
};class Derived : public Base {
public:~Derived() {std::cout << "Derived destructor called" << std::endl;}
};int main() {Base* obj = new Derived();  // 基类指针指向派生类对象delete obj;  // 销毁对象return 0;
}

输出

Derived destructor called
Base destructor called

在这个例子中,基类的析构函数被定义为虚函数,结果 delete 操作时,首先调用了派生类 Derived 的析构函数,然后调用基类 Base 的析构函数,确保了对象的正确销毁。这保证了派生类对象的所有资源,包括其特有的成员,都得到了正确的释放。

 详细说明

  • 虚函数表(vtable):当基类的析构函数被声明为虚函数时,C++编译器会为类创建一个虚函数表(vtable),这个表用于在运行时找到正确的析构函数。delete 操作会通过虚函数表找到派生类的析构函数,并依次调用析构函数链,从派生类到基类逐层销毁对象。

  • 避免资源泄漏:如果基类的析构函数不是虚函数,当基类指针或引用指向派生类对象时,销毁对象只会调用基类的析构函数,派生类的析构函数不会被执行,从而导致派生类中的资源(如动态分配的内存、文件句柄等)无法被释放,进而导致资源泄漏。

带有资源管理的派生类

class Base {
public:virtual ~Base() {std::cout << "Base destructor called" << std::endl;}
};class Derived : public Base {
private:int* data;
public:Derived() {data = new int[10];  // 动态分配内存std::cout << "Derived constructor: allocating memory" << std::endl;}~Derived() {delete[] data;  // 释放内存std::cout << "Derived destructor: releasing memory" << std::endl;}
};int main() {Base* obj = new Derived();delete obj;  // 销毁对象,正确调用析构函数链return 0;
}

输出

Derived constructor: allocating memory
Derived destructor: releasing memory
Base destructor called

在这个例子中,Derived 类动态分配了内存。如果基类的析构函数不是虚函数,那么在销毁对象时,Derived 的析构函数将不会被调用,导致动态分配的内存无法释放。而通过将基类的析构函数声明为虚函数,C++会确保派生类的析构函数被正确调用,从而释放所有资源。

什么时候不需要虚析构函数?

如果一个基类永远不会作为指针或引用来操作派生类对象,或者你确定它不会被继承,那么就不需要将析构函数声明为虚函数。例如:

  • 没有继承关系的类:如果类不是基类或者不会被继承,那么没有必要将析构函数设为虚函数。
  • 无需动态分配和多态的简单基类:如果一个基类永远不会被作为指针或引用使用,且派生类对象不会通过基类指针销毁,那么可以不使用虚析构函数。

总结

基类的析构函数定义为虚函数的主要原因是为了确保当通过基类指针或引用销毁派生类对象时,派生类的析构函数也能被正确调用,防止资源泄漏。虚析构函数的必要条件通常出现在使用多态动态内存分配时。

  • 必须使用虚函数的情况:当你需要通过基类指针或引用销毁对象时,基类的析构函数必须是虚函数。
  • 虚函数的作用:保证派生类的析构函数被调用,正确销毁派生类对象,释放所有资源。

7.多继承存在什么问题?如何消除多继承中的二义性?

在C++中,多继承允许一个类同时继承多个基类,这样可以使类获得多个基类的属性和行为。然而,多继承带来了几个潜在的问题,最常见的是二义性问题菱形继承问题。这些问题的存在会影响代码的可读性、维护性和正确性。

1. 二义性问题(Ambiguity Problem)

当一个派生类从多个基类继承,且这些基类中存在同名的成员函数或成员变量时,C++编译器会无法判断该调用哪个基类的成员,导致二义性

#include <iostream>
using namespace std;class Base1 {
public:void show() {cout << "Base1 show()" << endl;}
};class Base2 {
public:void show() {cout << "Base2 show()" << endl;}
};class Derived : public Base1, public Base2 {// Derived 类从 Base1 和 Base2 都继承了 show() 函数
};int main() {Derived d;d.show();  // 二义性错误:编译器无法确定调用 Base1::show() 还是 Base2::show()return 0;
}

问题解释: 在这个例子中,Derived 类从 Base1Base2 继承了同名的 show() 函数。编译器在遇到 d.show() 时无法确定应该调用 Base1show() 还是 Base2show(),因此会报二义性错误。

解决方法:显式指定基类

为了解决这个问题,必须通过作用域解析运算符显式指定要调用的基类成员。

int main() {Derived d;d.Base1::show();  // 调用 Base1::show()d.Base2::show();  // 调用 Base2::show()return 0;
}

输出

Base1 show()
Base2 show()

通过在调用时使用基类名加上作用域解析运算符 ::,我们可以明确告诉编译器希望调用哪个基类的成员函数,从而消除二义性。

2. 菱形继承问题(Diamond Problem)

菱形继承(又称钻石继承)是多继承中最典型的问题之一。它的结构类似于一个菱形:两个类继承自同一个基类,然后另一个派生类同时继承这两个子类。这样会导致重复继承,即派生类将有两份基类的拷贝。

菱形继承问题
#include <iostream>
using namespace std;class Base {
public:int data;Base() : data(0) {}
};class Derived1 : public Base {
};class Derived2 : public Base {
};class FinalDerived : public Derived1, public Derived2 {
};int main() {FinalDerived fd;// fd.data;  // 这会导致编译器报错,无法确定是 Derived1::Base 还是 Derived2::Base 中的 datareturn 0;
}

问题解释FinalDerived 类通过 Derived1Derived2 都继承了 Base 类,因此 FinalDerived 类有两份 Base 类的拷贝。当我们访问 fd.data 时,编译器无法判断应该访问 Derived1::Base 还是 Derived2::Base 中的 data,从而导致二义性。

解决方法:虚继承(Virtual Inheritance)

C++ 通过虚继承解决了菱形继承中的重复继承问题。虚继承确保只有一份基类的实例被继承,从而消除重复拷贝。

#include <iostream>
using namespace std;class Base {
public:int data;Base() : data(0) {}
};class Derived1 : public virtual Base {  // 使用虚继承
};class Derived2 : public virtual Base {  // 使用虚继承
};class FinalDerived : public Derived1, public Derived2 {
};int main() {FinalDerived fd;fd.data = 100;  // 现在只有一份 Base::datacout << fd.data << endl;  // 输出 100return 0;
}

输出

100

通过将 Derived1Derived2Base 进行虚继承,我们确保了 FinalDerived 类只有一份 Base 的成员变量 data,从而解决了菱形继承带来的二义性问题。

3. 二义性问题的其他解决方案

除了显式指定基类和虚继承外,还有其他一些解决多继承中二义性的方法:

(1) 避免多继承

如果可以通过设计优化避免多继承,通常是最好的选择。比如,可以通过组合(composition)或接口来代替多继承。这种方式能让设计更加清晰,避免多继承带来的复杂性。

(2) 使用接口类(抽象类)

在某些场景下,使用接口类(只包含纯虚函数的类)可以解决二义性问题。这种方式类似于Java中的接口,允许类实现多个接口,而不会带来二义性问题。

class Interface1 {
public:virtual void show() = 0;  // 纯虚函数
};class Interface2 {
public:virtual void show() = 0;  // 纯虚函数
};class Derived : public Interface1, public Interface2 {
public:void show() override {cout << "Derived show()" << endl;}
};int main() {Derived d;d.show();  // 二义性问题不存在,因为 Derived 类重写了所有纯虚函数return 0;
}

在这个例子中,虽然 Interface1Interface2 中都有 show() 函数,但由于它们是纯虚函数,Derived 类必须实现 show(),因此不会产生二义性。

4. 总结

  • 多继承的潜在问题

    • 二义性问题:当多个基类有同名成员时,派生类无法明确调用哪个基类的成员。
    • 菱形继承问题:多个派生类从同一个基类继承,导致最终派生类拥有多个基类的实例。
  • 解决方案

    1. 显式指定基类:通过作用域解析符明确指出要调用哪个基类的成员。
    2. 虚继承:通过虚继承避免菱形继承导致的重复基类实例。
    3. 组合和接口:使用组合(composition)或接口(抽象类)设计模式来替代多继承,简化设计,避免复杂的继承结构。

8.拷贝构造函数赋值运算符重载

拷贝构造函数赋值运算符重载在C++中都是用于处理对象的复制,但它们的使用场景和行为有所不同。让我们详细分析这两者的定义区别以及它们各自的应用场景

1. 拷贝构造函数(Copy Constructor)

拷贝构造函数用于创建对象时的初始化,即用一个已经存在的对象来初始化新创建的对象。它的调用发生在对象创建的同时,用另一个对象作为副本进行初始化。

定义:
class MyClass {
public:MyClass(const MyClass& other);  // 拷贝构造函数
};
触发拷贝构造函数的场景:
  • 当一个对象直接初始化另一个对象时,例如:
    MyClass obj1;
    MyClass obj2 = obj1;  // 调用拷贝构造函数
    
  • 将对象作为参数传递给函数时(按值传递):
    void func(MyClass obj);  // 传递对象时调用拷贝构造函数
    
  • 当函数返回一个对象(按值返回)时:
    MyClass func() {MyClass temp;return temp;  // 返回对象时调用拷贝构造函数
    }
    

2. 赋值运算符重载(Assignment Operator Overload)

赋值运算符重载用于将一个已经存在的对象的值赋给另一个已经存在的对象。它发生在两个对象都已经创建之后,进行值的赋值操作。

定义:
class MyClass {
public:MyClass& operator=(const MyClass& other);  // 赋值运算符重载
};
触发赋值运算符重载的场景:
  • 当一个已经存在的对象被赋值为另一个对象时,例如:
    MyClass obj1;
    MyClass obj2;
    obj2 = obj1;  // 调用赋值运算符
    
  • 赋值发生时,目标对象必须已经存在。

3. 区别总结

比较点拷贝构造函数赋值运算符重载
目的用一个已有对象创建新对象将一个已有对象的内容赋值给另一个已有对象
调用时机对象创建时,用另一个对象初始化对象已经存在,然后将另一个对象的值赋给它
函数签名MyClass(const MyClass& other)MyClass& operator=(const MyClass& other)
内存分配可能会分配新内存给新对象通常不会分配新内存(除非需要深拷贝)
默认行为逐成员拷贝(浅拷贝)逐成员赋值(浅拷贝)
操作对象的数量创建一个新的对象操作两个已经存在的对象
自引用不涉及自引用可能需要检查自引用
返回类型无返回值返回对当前对象的引用(*this

4. 默认实现 vs 自定义实现

C++ 自动为每个类提供默认的拷贝构造函数和赋值运算符,但它们都是执行浅拷贝,即逐成员的复制或赋值。如果类中包含动态分配的内存或其他需要深度管理的资源,必须自定义拷贝构造函数和赋值运算符,以确保正确处理这些资源。

如果类包含动态内存分配,使用默认的拷贝和赋值可能会导致资源管理问题(如重复释放内存、悬空指针等)。

5. 深拷贝示例

对于包含动态内存的类,需要自定义拷贝构造函数和赋值运算符以执行深拷贝,确保在复制对象时分配独立的内存,而不是简单复制指针。

class MyClass {
private:int* data;  // 动态分配的资源
public:// 构造函数MyClass(int value) : data(new int(value)) {}// 拷贝构造函数(深拷贝)MyClass(const MyClass& other) : data(new int(*other.data)) {std::cout << "Copy constructor called" << std::endl;}// 赋值运算符重载(深拷贝)MyClass& operator=(const MyClass& other) {std::cout << "Assignment operator called" << std::endl;if (this == &other)  // 防止自我赋值return *this;// 先释放当前对象的资源delete data;// 分配新的资源并复制数据data = new int(*other.data);return *this;}// 析构函数~MyClass() {delete data;}// 打印数据void print() const {std::cout << "Value: " << *data << std::endl;}
};int main() {MyClass obj1(10);MyClass obj2 = obj1;  // 调用拷贝构造函数obj2.print();MyClass obj3(20);obj3 = obj1;  // 调用赋值运算符obj3.print();return 0;
}

输出

Copy constructor called
Value: 10
Assignment operator called
Value: 10

6. 自我赋值问题

在实现赋值运算符时,必须处理自我赋值(self-assignment)的情况。自我赋值指的是对象将自己赋值给自己,例如 obj = obj;。如果不处理这种情况,可能导致内存泄漏或其他不良行为。

在赋值运算符的实现中,一般通过检查 this 指针来避免自我赋值:

if (this == &other) {return *this;
}

7. 总结

  • 拷贝构造函数用于在创建对象时通过另一个对象进行初始化。
  • 赋值运算符重载用于将一个对象的值赋给另一个已存在的对象。
  • 二者的最大区别在于调用时机,前者是在创建新对象时,后者是在赋值时
  • 如果类涉及动态资源管理(如动态内存分配),需要自定义深拷贝的拷贝构造函数和赋值运算符,并处理自我赋值问题。

9.类型转换分为哪几种?各自有什么样的特点?

在C++中,类型转换(Type Casting)分为隐式类型转换显式类型转换两大类。显式类型转换还可以通过C风格的强制转换C++提供的四种类型转换操作符来实现。

1. 隐式类型转换(Implicit Type Conversion)

隐式类型转换又称为自动类型转换,是指编译器自动将一种数据类型转换为另一种兼容类型,不需要程序员明确地写出转换语句。

特点:
  • 自动进行:不需要程序员干预,编译器会根据需要自动转换类型。
  • 类型兼容性:通常发生在兼容类型之间,比如从低精度类型向高精度类型转换(如 int 转换为 double),或从窄类型转换为宽类型(如 char 转换为 int)。
  • 数据可能丢失:在某些情况下可能导致数据精度丢失,比如将 double 转换为 int 时,小数部分会被截断。
int x = 10;
double y = x;  // 隐式类型转换:int 转换为 double

在这个例子中,int 类型的 x 自动转换为了 double 类型,并赋值给 y

2. 显式类型转换(Explicit Type Conversion)

显式类型转换,也叫强制类型转换,要求程序员明确地指定数据类型转换,常见的方式有C风格的强制转换和C++的四种类型转换操作符。

2.1 C风格强制转换(C-style Cast)

C风格的类型转换语法类似于函数调用,通过在类型前加上目标类型的括号实现强制转换。

特点:
  • 语法简单:C风格类型转换语法简单。
  • 不安全:由于它忽略了类型安全检查,容易引发问题,尤其在复杂对象和指针的转换中。
int a = 10;
double b = (double)a;  // C风格的强制转换
2.2 C++类型转换操作符

C++引入了四种类型转换操作符,提供了更严格和明确的类型转换机制,主要包括:

  • static_cast:用于大多数标准转换
  • dynamic_cast:用于安全地向下转换多态类型
  • const_cast:用于移除或添加const属性
  • reinterpret_cast:用于低级别、危险的指针转换
2.2.1 static_cast

static_cast 是最常用的类型转换操作符,用于执行任何可以在编译时检查的转换。适用于基本数据类型之间的转换、指针类型之间的转换(前提是指针类型兼容)等。

特点:
  • 编译时转换:在编译阶段进行转换,编译器会进行类型检查。
  • 安全性较高:不会像 reinterpret_cast 那样进行完全不同类型之间的转换,确保类型转换是合法的。
int a = 10;
double b = static_cast<double>(a);  // 用 static_cast 进行类型转换
2.2.2 dynamic_cast

dynamic_cast 用于在继承体系中进行安全的向下转换(downcasting)。它要求基类中至少有一个虚函数(通常是虚析构函数),以确保对象具有多态性。

特点:
  • 运行时检查dynamic_cast 会在运行时检查转换是否安全。如果转换失败,指针会返回 nullptr,引用会抛出 std::bad_cast 异常。
  • 适用于多态类型:只能用于具有多态性的类,即类中包含虚函数。
class Base {
public:virtual void show() {}
};class Derived : public Base {
public:void show() override {}
};Base* basePtr = new Derived();
Derived* derivedPtr = dynamic_cast<Derived*>(basePtr);  // 安全的向下转换
if (derivedPtr) {derivedPtr->show();
} else {std::cout << "Conversion failed!" << std::endl;
}
2.2.3 const_cast

const_cast 用于移除或添加const限定,常用于对常量对象进行修改或对非const指针进行转换。

特点:
  • 仅限于const属性的移除或添加:不能用于其他类型的转换,只能用于指针或引用的const属性操作。
  • 不改变底层类型:它不会改变对象的底层类型,只是改变了对象的const属性。
const int a = 10;
int* p = const_cast<int*>(&a);  // 移除 const 限定
*p = 20;  // 可能引发未定义行为
2.2.4 reinterpret_cast

reinterpret_cast 是一种非常底层的强制类型转换,用于将一种类型的指针转换为另一种不相关类型的指针,或者转换为整数。它允许进行位级别的转换,因此是最不安全的转换方式。

特点:
  • 危险且不安全reinterpret_cast 可以将不相关的指针类型相互转换,编译器不进行类型安全检查,容易导致运行时错误。
  • 用于低级别转换:通常在需要进行底层数据操作时使用。
int a = 42;
int* p = &a;
char* cp = reinterpret_cast<char*>(p);  // 将 int* 转换为 char*

3. 各类类型转换的特点总结

类型转换方式特点使用场景
隐式类型转换编译器自动进行,安全性较高兼容类型之间的自动转换,如 int 转换为 double
C风格强制转换简单但不安全,容易产生问题不推荐在C++中使用
static_cast编译时转换,适用于大部分标准转换,安全性较高基本类型之间的转换、兼容指针类型之间的转换
dynamic_cast运行时检查,多用于继承体系中的向下转换,依赖多态性需要安全的向下转换多态类时
const_cast用于移除或添加const限定需要修改const对象或指针的const属性时
reinterpret_cast最危险的类型转换,允许进行不同类型的强制转换,编译器不检查低级别数据操作,进行位级别转换时

4. 总结

  • 隐式类型转换:编译器自动进行,常见于兼容类型之间。
  • C风格强制转换:语法简单但不推荐使用,C++提供了更安全的类型转换操作符。
  • static_cast:适用于标准类型转换,安全性较高。
  • dynamic_cast:适用于继承体系中的多态转换,确保安全性。
  • const_cast:用于const属性的移除或添加。
  • reinterpret_cast:危险的底层转换,通常在位级别操作中使用。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/458957.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深度学习:YOLO v1网络架构、损失值及NMS极大值抑制

引言 随着深度学习的发展&#xff0c;物体检测&#xff08;Object Detection&#xff09;成为计算机视觉领域的一项重要任务。传统的物体检测方法往往依赖于手工设计的特征和滑窗搜索策略&#xff0c;效率低下且效果有限。近年来&#xff0c;基于深度学习的方法&#xff0c;尤…

leetcode-63-不同陆路径II

题解&#xff1a; 1、设dp[i][j]为到达(i,j)点的路径。当grid[i][j]1时,dp[i][j]0;否则dp[i][j]为到达(i-1,j)的最多路径与到达(i,j-1)的最多路径之和。当(i,j)位于第一行时&#xff0c;dp[i][j]dp[i][j-1]。当(i,j)位于第一列时&#xff0c;dp[i][j]dp[i-1][j]。 2、初始化M…

MATLAB锂电概率分布模型

&#x1f3af;要点 概率分布等效电路模型结合了路径相关速率能力及状态估计中滞后效应。纠正了充电状态中时间误差累积及避免开路电压中电压滞后现象。使用电流方向和电池容量相关函数描述开路电压&#xff0c;并使用微分方程描述电压滞后现象。模型结构基于一级相变的材料机制…

新华三H3CNE网络工程师认证—OSPF路由协议

OSPF是典型的链路状态路由协议&#xff0c;是目前业内使用非常广泛的IGP协议之一。本博客将对OSPF路由协议进行总结。 OSPF目前针对IPv4协议使用的是OSPFVersion2(RFC2328)&#xff1b; 针对IPv6协议使用OSPFVersion3(RFC2740)。如无特殊说明本章后续所指的OSPF均为OSPF Versi…

监督学习之逻辑回归

逻辑回归&#xff08;Logistic Regression&#xff09; 逻辑回归是一种用于二分类&#xff08;binary classification&#xff09;问题的统计模型。尽管其名称中有“回归”二字&#xff0c;但逻辑回归实际上用于分类任务。它的核心思想是通过将线性回归的输出映射到一个概率值…

【MATLAB源码-第193期】基于matlab的网络覆盖率NOA优化算法仿真对比VFINOA,VFPSO,VFNGO,VFWOA等算法。

操作环境&#xff1a; MATLAB 2022a 1、算法描述 NOA&#xff08;Network Optimization Algorithm&#xff0c;网络优化算法&#xff09;是一个针对网络覆盖率优化的算法&#xff0c;它主要通过优化网络中节点的分布和配置来提高网络的整体覆盖性能。网络覆盖率是衡量一个无…

基于STM32G0的USB PD协议学习(3)

0、前言 STM32这个平台资源确实很不错&#xff0c;但是里面的PD代码是一个lib库文件&#xff0c;没有开源。可以做来玩玩&#xff0c;但是如果要满足USB-IF认证需求的话&#xff0c;谨慎&#xff01;&#xff01;&#xff01; 这段时间较为繁忙&#xff0c;断更有点严重... …

植物健康,Spring Boot来保障

5系统详细实现 5.1 系统首页 植物健康系统需要登录才可以看到首页。具体界面的展示如图5.1所示。 图5.1 系统首页界面 5.2 咨询专家 可以在咨询专家栏目发布消息。具体界面如图5.2所示。 图5.2 咨询专家界面 5.3 普通植物检查登记 普通员工可以对普通植物检查登记信息进行添…

Linux之权限(2)

权限&#xff08;2&#xff09; 操作&#xff1a;有VS没有 只有文件的拥有者或者root&#xff0c;能修改自己的权限 Linux下能执行真的是一个可执行文件可执行权限 user,group,other和我自己进行身份对比&#xff0c;依次只对比一次 8进制数值表示方法 chmod ax /home/abc.…

基于vue、VantUI、django的程序设计

首先构建vue项目&#xff0c;构建项目点这里 安装 npm install axios axios简介 Axios 是一个基于 promise 的 HTTP 库&#xff0c;用于发起请求和接收响应&#xff0c;实现异步操作 基本使用 axios对象 请求响应拦截 在utils文件夹里新建ajax.js 创建一个axios对象并…

云智慧完成华为原生鸿蒙系统的适配, 透视宝 APM 为用户体验保驾护航

2024 年 10 月 22 日&#xff0c;首个国产移动操作系统 —— 华为原生鸿蒙操作系统 HarmonyOS NEXT 正式面世&#xff0c;成为继 iOS 和 Android 后的全球第三大移动操作系统。HarmonyOS NEXT&#xff0c;从系统内核、数据库根基&#xff0c;到编程语言创新、AI&#xff08;人工…

【WebGis开发 - Cesium】三维可视化项目教程---图层管理拓展图层透明度控制

目录 引言一、为什么要开发图层透明度控制功能二、开发思路整理1. cesium图层api查询1.1 imageryLayer 透明度1.2 primitive 透明度 三、代码编写1. 修改原有图层管理代码2. 新增页面结构3. 编写图层透明度控制方法 四、总结 引言 本教程主要是围绕Cesium这一开源三维框架开展的…

如何通过sip信令以及抓包文件分析媒体发到哪个地方

前言 问题描述&#xff1a;A的媒体没转发到B&#xff0c;B只能听到回铃音&#xff0c;没有A的说话声音&#xff0c;并且fs这边按正常的信令发送了. 分析流程 分析早期媒体发送到哪一个IP 10.19.0.1发送了一个请求给10.19.0.157这个IP&#xff0c;然而这里的SDP媒体地址&am…

react 总结+复习+应用加深

文章目录 一、React生命周期1. 挂载阶段&#xff08;Mounting&#xff09;补充2. 更新阶段&#xff08;Updating&#xff09;补充 static getDerivedStateFromProps 更新阶段应用补充 getSnapshotBeforeUpdate3. 卸载阶段&#xff08;Unmounting&#xff09; 二、React组件间的…

搭建 mongodb 副本集,很详细

搭建 mongodb 副本集&#xff0c;很详细 一、前言二、创建用户1、创建 root 用户2、创建测试用户3、修改用户密码 三、修改配置文件&#xff08;主节点&#xff09;1、开启登录认证2、加上副本集3、最终配置文件 四、副本节点1、创建副本节点目录2、编辑配置文件3、启动副本节点…

2024年四川省大学生程序设计竞赛 补题记录

文章目录 Problem A. 逆序对染色&#xff08;思维树状数组&#xff09;Problem B. 连接召唤&#xff08;贪心&#xff09;Problem E. L 型覆盖检查器&#xff08;模拟&#xff09;Problem F. 小球进洞&#xff1a;平面版&#xff08;几何&#xff09;Problem G. 函数查询Proble…

关于传输线电感

要理解自感、互感、回路电感&#xff0c;PCB表层单位长度电感约为7.5纳亨每英寸&#xff0c;内层约为9纳亨每英寸 磁力线 电流周围会产生磁场&#xff0c;可以认为磁场是由许多“力线”构成。 电流穿过与其垂直的某一平面时&#xff0c;在该平面内激起一个“磁漩涡” 形成许多…

JavaSet集合

无序&#xff08;指的是添加顺序和获取出的数据顺序不一致&#xff0c;不重复&#xff0c;无索引 既然Set没有索引&#xff0c;因此功能同上一篇Connection的功能&#xff0c;几乎没有额外的功能 HashSet的原理 为什么是无序&#xff08;要构建红黑树&#xff09;&#xff0…

【Linux学习】(9)调试器gdb

前言 Linux基础工具&#xff1a;安装软件我们用的是yum&#xff0c;写代码用的是vim&#xff0c;编译代码用gcc/g&#xff0c;调试代码用gdb&#xff0c;自动化构建用make/Makefile&#xff0c;多人协作上传代码到远端用的是git。 在前面我们把yum、vim、gcc、make、git都已经学…

逆向工程基本概念

引言 逆向工程&#xff08;Reverse Engineering&#xff09;是指从已经存在的产品或系统中提取信息&#xff0c;并理解其设计原理的过程。在软件开发中&#xff0c;逆向工程通常用于理解一个已有软件系统的内部工作原理&#xff0c;可能是为了兼容性、安全分析、修复或者改进等…