承德市建设工程交易中心网站,花木网站模版,修改备案网站信息,请人做网站域名和主机c的几种构造函数 构造函数拷贝构造函数转换构造函数移动构造函数 析构函数 构造函数
C中的构造函数可以分为5类#xff1a;默认构造函数、普通构造函数、拷贝构造函数、转换构造函数、移动构造函数。
好像还有委托构造
默认构造和普通构造和java基本一样
详细
拷贝构造函… c的几种构造函数 构造函数拷贝构造函数转换构造函数移动构造函数 析构函数 构造函数
C中的构造函数可以分为5类默认构造函数、普通构造函数、拷贝构造函数、转换构造函数、移动构造函数。
好像还有委托构造
默认构造和普通构造和java基本一样
详细
拷贝构造函数
拷贝构造函数用于从另一个已存在的对象创建新对象。 而不共享原始对象的数据。
class Person {
public:string name;int age;Person(const Person other) { // 拷贝构造函数name other.name;age other.age;}
};int main() {Person p1(Alice, 30); // 创建 Person 对象Person p2(p1); // 拷贝构造函数创建 p2Person p3p1;//!!!这样也会调用拷贝构造函数这是全新的对象 和java不一样cout p2.name , p2.age endl; // 输出Alice, 30return 0;
}如果你没有显式定义拷贝构造函数编译器会自动生成一个默认拷贝构造函数。这个默认的拷贝构造函数会逐成员地对对象进行浅拷贝。
补充
Complex c2(c1);
Complex c2 c1;
//这两条语句是等价的。复制构造函数的调用Complex c1, c2;
c1c2;
//第二条语句是初始化语句不是赋值语句。不会调用复制构造函数
//此时是对 c1 赋予新的值 c2这会调用类的拷贝赋值运算符operator()
//在 C 中c1 c2; 是深拷贝行为。c1 和 c2 是两个独立的对象。c1 c2; 将 c2 的值拷贝到 c1相当于复制了 c2 的内容。还有补充
class A{
public:A(){};A(A a){coutCopy constructor calledendl;}
};
void Func(A a){ }
int main(){A a;Func(a); //这里会调用A类的拷贝构造函数return 0;
}如果形参是一个对象那么形参的值是否等于实参取决于该对象所属的类的复制构造函数是如何实现的。如果复制构造函数随便写的那传进去的就…
在函数被调用时生成的形参要用复制构造函数初始化这会带来时间上的开销。可以用将形参声明为对象的 const 引用代替
转换构造函数
一个构造函数接收一个不同于其类类型的形参可以视为将其形参转换成类的一个对象。像这样的构造函数称为转换构造函数。在 C string 类中可以找到使用转换构造函数的实用示例。string 类提供一个将 C 字符串转换为 string 的转换构造函数
class string
{//仅显示转换构造函数public:string(char *);//形参时其他类型变量且只有一个形参
};移动构造函数
移动构造和move 建议这一篇我的笔记
下面这个例子是没有使用移动构造函数的会有内存的浪费
// g 14_Copy_Date.cpp -stdc11
#include iostream
#include stdio.h
#include string.husing namespace std;#define MAX_NEW_MEM (64*1000*1000)class CDate
{
public:CDate(int year, int mon, int day); // 构造函数声明CDate(const CDate date); // 拷贝构造函数声明~CDate(); // 析构函数声明CDate operator(int day); // 加号运算符声明void show(){cout Date: m_year . m_mon . m_day , this this endl;//cout Date: str endl;}private:int m_year;int m_mon;int m_day;char *str;
};// 构造函数定义
CDate::CDate(int year, int mon, int day)
{m_year year;m_mon mon;m_day day;str new char[MAX_NEW_MEM];sprintf(str, %4d.%02d.%02d, year,mon,day);cout Calling Constructor , this this endl;
}// 拷贝构造函数定义
CDate::CDate(const CDate date)
{m_year date.m_year;m_mon date.m_mon;m_day date.m_day;str new char[MAX_NEW_MEM];memcpy(str, date.str, MAX_NEW_MEM);cout Calling Copy Constructor , this this , Copy Data endl;
}// 析构函数定义
CDate::~CDate()
{cout Calling Destructor , this this endl;delete [] str;
}CDate CDate::operator(int day)
{CDate temp *this;temp.m_day day;cout Calling operator , this temp endl;return temp;
}int main()
{CDate date(2024,06,07);cout endl;CDate date1 std::move(date1);// std::move 强制将 date1 的求值结果转为右值//std::move 强制将 date1 的求值结果转为右值避免编译器优化//这里其实 std::move(date1) 是temp//temp 对象赋值给date1之后就销毁了如果可以直接将temp对象的资源给到date1就可以减少一次复制。date1.show();cout endl;return 0;
} 参考
移动构造
CDate(CDate date) noexcept; // 声明
CDate::CDate(CDate date) noexcept // 实现 加上noexcept用于通知标准库不抛出异常。提高性能
{m_year date.m_year;m_mon date.m_mon;m_day date.m_day;str date.str;date.str NULL;cout Calling Move Constructor , this this endl;
}
int main()
{CDate date(2024,06,07);CDate date1 std::move(date1);// std::move 强制将 date1 的求值结果转为右值date1.show();return 0;
}运行结果可以看到相比于第一小节的代码这里调用了移动构造函数减少了一次拷贝。 std::move
class B
{
public:B() {}B(const B) { cout B Copy Constructor endl; }
};class A {
private:B* pb;public:// 默认构造函数A() {pb new B();cout A Constructor endl;}// 拷贝构造函数深拷贝A(const A src) {pb new B(*(src.pb));cout A Copy Constructor endl;}// 移动构造函数A(A src) noexcept {pb src.pb;src.pb nullptr; // 防止 src 析构时释放原来的堆内存cout A Move Constructor endl;}// 析构函数~A() {delete pb;cout A Destructor endl;}
};static A getA()
{A a;cout endl;return a;
}int main()
{A a getA();//A a getA();调用的是A的移动构造cout endl;A a1(a);//A a1(a); 调用的是A的拷贝构造。A的拷贝构造需要对成员变量B进行深拷贝cout endl;A a2(std::move(a));//A a2(std::move(a));将a转换为右值因此a2调用的是移动构造而不是拷贝构造。system(pause);return 0;
}析构函数
它会在每次删除所创建的对象时执行。
析构函数的名称与类的名称是完全相同的只是在前面加了个波浪号~作为前缀