c++中的构造函数有哪些类型_构造函数的分类与使用区别解析

C++中构造函数包括默认、带参、拷贝、移动和委托五种类型。默认构造函数在无参数时调用,用于初始化对象;带参构造函数实现定制化初始化;拷贝构造函数通过已有对象创建新对象,常用于值传递;移动构造函数利用右值引用提升性能,避免深拷贝;委托构造函数减少代码重复,支持构造函数间调用。合理使用可提升类的安全性与效率,并需遵循三法则或五法则以正确管理资源。

在C++中,构造函数是类的一种特殊成员函数,用于初始化对象。它在创建对象时自动调用,确保对象的成员变量被正确设置。根据不同的使用场景和初始化方式,构造函数可以分为多种类型。下面介绍常见的构造函数类型及其使用区别。

1. 默认构造函数(Default Constructor)

默认构造函数是指不需要参数就能调用的构造函数。它可以是无参构造函数,也可以是所有参数都有默认值的构造函数。

特点:

  • 在定义对象但不提供参数时自动调用。
  • 如果类中没有显式定义任何构造函数,编译器会自动生成一个默认构造函数(不执行任何操作)。
  • 一旦用户定义了任意构造函数,编译器将不再生成默认构造函数。
示例:
class Person {
public:
    Person() {
        name = "Unknown";
        age = 0;
    }
private:
    string name;
    int age;
};
Person p; // 调用默认构造函数

2. 带参数的构造函数(Parameterized Constructor)

带参数的构造函数允许在创建对象时传入初始值,从而实现对成员变量的定制化初始化。

用途:

  • 用于初始化具有特定值的对象。
  • 避免对象创建后还需手动赋值。
示例:
class Person {
public:
    Person(string n, int a) {
        name = n;
        age = a;
    }
private:
    string name;
    int age;
};
Person p("Alice", 25); // 调用带参构造函数

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

拷贝构造函数用于通过已存在的对象创建新对象,其参数是同类对象的引用。

调用时机:

  • 用一个对象初始化另一个对象:Person p2(p1);
  • 函数传参时以值传递方式传入对象。
  • 函数返回对象时以值返回。

若未定义,编译器会生成默认的拷贝构造函数,执行浅拷贝。对于包含指针或动态资源的类,通常需要自定义深拷贝逻辑。

示例:
class Person {
public:
    Person(const Person& other) {
        name = other.name;
        age = other.age;
    }
};
Person p1("Bob", 30);
Person p2 = p1; // 调用拷贝构造函数

4. 移动构造函数(Move Constructor)

C++11引入了右值引用和移动语义,移动构造函数用于“窃取”临时对象的资源,避免不必要的深拷贝,提升性能。

特点:

  • 参数为右值引用:T(T&& other)
  • 适用于临时对象(如函数返回值、std::move的结果)。
  • 常用于管理动态内存、文件句柄等资源的类。
示例:
class Buffer {
public:
    Buffer(Buffer&& other) noexcept {
        data = other.data;
        size = other.size;
        other.data = nullptr; // 防止原对象释放资源
        other.size = 0;
    }
private:
    int* data;
    size_t size;
};
Buffer b1 = createBuffer(); // 可能调用移动构造函数

5. 委托构造函数(Delegating Constructor)

C++11支持构造函数之间的相互调用,即一个构造函数可以调用类中的另一个构造函数,减少代码重复。

示例:
class Person {
public:
    Person() : Person("", 0) {}                    // 委托给带参构造
    Person(string n) : Person(n, 0) {}             // 委托
    Person(string n, int a) : name(n), age(a) {}
private:
    string name;
    int age;
};

使用区别总结

选择依据:

  • 需要无参初始化 → 使用默认构造函数。
  • 需传入具体值 → 使用带参构造函数。
  • 基于已有对象创建新对象 → 拷贝构造函数发挥作用。
  • 提高性能,避免复制大对象 → 实现移动构造函数。
  • 多个构造函数逻辑相似 → 使用委托构造减少重复。

基本上就这些。合理使用各类构造函数,能让类的设计更安全、高效、易用。注意在涉及资源管理时,遵循“三法则”或“五法则”,确保拷贝与移动行为正确。