初始化:
当对象在创建时获取了一个特定的值,我们就说该对象被初始化了。
在C++语言中,初始化是一个异常复杂的问题,我们也将反复讨论这个问题。很多程序员对使用=来初始化的方式感到困惑。这种方式容易让人认为初始化是赋值的一种,事实上在C++语言中,初始化和赋值是两种完全不同的操作。然而在很多编程语言中,这两者的区别完全可以忽略不计,即使在C++语言中有时这两者区别也无关紧要,所以人们特别容易将二者混为一谈。需要强调的是,这个概念至关重要。
初始化不是赋值,初始化的含义是在创建对象时赋予一个初值,而赋值是将对象的当前值擦除掉,以一个新值代替。
什么时候是对象的初始化,最简单的判断方式是:当有对象创建的时候,就是初始化,未创建对象而对象的值被改变就是赋值。
C++语言的初始化方式
1、直接初始化
2、拷贝初始化
我们来看看C++Primer中的说法:
“当用于类类型对象时,初始化的拷贝形式和直接形式有所不同:直接初始化直接调用与实参匹配的构造函数,拷贝初始化总是调用拷贝构造函数。拷贝初始化首先使用指定构造函数创建一个临时对象,然后用拷贝构造函数将那个临时对象拷贝到正在创建的对象。“
从上面的说法中,我们可以知道,直接初始化不一定调用拷贝构造函数(除非传入的实参为相同类对象实例)。拷贝初始化是一定要调用拷贝构造函数。
直接初始化与拷贝初始化区别方式,在对象初始化时,使用"="的为拷贝初始化,其他为直接初始化。
class Test
{
public:
Test()
{
c[0] = '\0';
cout << "Test()" << endl;
}
Test(const char *pc)
{
strcpy(c, pc);
cout << "Test(const char *pc)" << endl;
}
Test& operator = (const Test &rhs)
{
strcpy(c, rhs.c);
cout << "Test& operator = (const Test &rhs)" << endl;
return *this;
}
Test(const Test& rhs)
{
strcpy(c, rhs.c);
cout << "Test(const Test& rhs)" << endl;
}
private:
char c[1024];
};
int main()
{
cout << "ct1: ";
Test ct1("ab"); //直接初始化
cout << "ct2: ";
Test ct2 = "ab"; //拷贝初始化
cout << "ct3: ";
Test ct3 = ct1; //拷贝初始化
cout << "ct4: ";
Test ct4(ct1); //直接初始化
cout << "ct5: ";
Test ct5 = Test(); //拷贝初始化
ct5 = ct1; //赋值操作
}
首先,上述类对象都是在创建对象时赋予一个新值,根据我们上面所说,ct1~ct5都是初始化,
ct1为直接初始化,直接调用Test(const char *pc)构造函数,匹配const char *pc;
ct2为拷贝初始化,先使用“ab”作为构造函数参数创建一个临时的Test对象,然后使用拷贝构造函数初始化ct2(注:此处在vs编译器会被优化为直接调用Test(const char *pc)构造函数。
ct3为拷贝初始化,使用拷贝构造函数Test(const Test &rhs)初始化ct3。
ct4为直接初始化,使用拷贝构造函数Test(const Test &rhs)初始化ct4。注意,直接初始化也可能使用拷贝构造函数,而拷贝构造函数必定是使用拷贝构造函数。
ct5为拷贝初始化,首先使用Test()创建一个临时的Test对象,再使用拷贝构造函数Test(const Test &rhs)初始化ct5
ct5 = ct1为赋值操作,因为ct5对象已经创建,此处仅仅是擦除掉ct5的值,而赋予一个新值。