首先,我们应该明确的是在C++中初始化不是赋值,因为初始化是必要的,如果读取了未初始化的值将会导致不明确的行为。初始化指创建变量并且给它赋初值,而赋值则是擦除对象的当前值并用新值代替。C++支持两种初始化变量的方式:复制初始化和直接初始化:
int ival(1000);//直接初始化是将初始化式放在括号里
int ival=1000;//复制初始化是用等号(=)
那么这两种方式有什么区别呢?我们可以这样认为,对于内置类型来说,复制初始化和直接初始化几乎没有差别,对于类类型来讲,当创建类类型对象时,初始化的复制形式和直接形式有所不同:直接初始化直接调用与实参匹配的构造函数,复制初始化总是调用复制构造函数。复制初始化首先使用指定构造函数创建一个临时对象,然后使用复制构造函数将那个临时对象复制到正在创建的对象:
string strVal1="2014";//编译器首先调用接受一个字符串形参的string构造函数,创建一个临时对象,然后,编译器使用
string复制构造函数将strVal1初始化为那个临时对象的副本
string strVal2("2014");//直接调用string类中匹配的构造函数
string strVal3 = string();//先使用默认构造函数创建一个临时对象,使用复制构造函数将那个临时对象复制到正在创建的对象
string strVal4;//直接运行strVal4的默认构造函数
其次,讲讲变量的初始化规则:
内置类型变量的初始化是否自动初始化取决于变量定义的位置,在函数体外定义的变量都初始化成0,在函数体内定义的内置变量不进行自动初始化,因此,实际编程中,建议每个内置类型的对象都要初始化,虽然有时候这样做并不总是必须的。类类型的初始化与它的默认构造函数是否存在有关,只要它有默认构造函数,则不管变量在那里定义,我们都可以不提供初始化式,相反,我们则需要每个定义都必须提供显式的初始化式,如:
string emptyStr;//因为类中定义了默认构造函数来初始化string变量为空字符串
接下来,我们来看看关于类类型的两种构造函数的写法
法一:
TestOne::TestOne(const std::string &name,const::string & address,
const std::list<PhoneNumber>& phones)
{
theName=name;
theAddress=address;//注意这些都是赋值,而非初始化
thePhones=phones;
}
法二:
TestTwo::TestTwo(const std::string &name,const::string & address,
const std::list<PhoneNumber>& phones)
:theName(name),
theAddress(address),//现在这些都是初始化
thePhones(phones)
{ }//现在,构造函数体不必有任何动作
这两种方法都可以满足我们的需要,但是效果是不一样的。
C++中规定,对象的成员变量的初始化动作发生在进入构造函数本体之前,在法一中,theName,theAddress,thePhone这些成员的初始化发生的时间是在这些成员的默认构造函数被自动调用之时,也就是它们真正被初始化的时候,因此,它们发生的时间比进入当前构造函数的时间更早。因此,法一相当于对这些已经被默认构造函数初始化的成员变量再次进行赋值,默认构造函数做了无用功。而法二,则是直接就通过成员的构造函数来初始化的,一步到位。