您的位置:首页 > 编程语言 > C语言/C++

C++类中各种不同类型成员根据是否static 、是否const类型的初始化方法不尽相同,写代码时经常搞乱,网上搜了一下资料,全部总结一下。一、例子 [cpp] view plaincopy ---

2016-03-18 15:25 549 查看


C++ 类中特殊的成员变量(常变量、引用、静态)的初始化方法

有些成员变量的数据类型比较特别,它们的初始化方式也和普通数据类型的成员变量有所不同。这些特殊的类型的成员变量包括:
a.引用
b.常量
c.静态
d.静态常量(整型)
e.静态常量(非整型)
常量和引用,必须通过参数列表进行初始化。

静态成员变量的初始化也颇有点特别,是在类外初始化且不能再带有static关键字,其本质见文末。
参考下面的代码以及其中注释:

#include <iostream>

using namespace std;
class BClass

{

public:

BClass() : i(1), ci(2), ri(i){} // 对于常量型成员变量和引用型成员变量,必须通过参数化列表的方式进行初始化

//普通成员变量也可以放在函数体里,但是本质其实已不是初始化,而是一种普通的运算操作-->赋值运算,效率也低

private:

int i; // 普通成员变量

const int ci; // 常量成员变量

int &ri; // 引用成员变量

static int si; // 静态成员变量

//static int si2 = 100; // error: 只有静态常量成员变量,才可以这样初始化

static const int csi; // 静态常量成员变量

static const int csi2 = 100; // 静态常量成员变量的初始化(Integral type) (1)

static const double csd; // 静态常量成员变量(non-Integral type)

//static const double csd2 = 99.9; // error: 只有静态常量整型数据成员才可以在类中初始化

};
//注意下面三行:不能再带有static

int BClass::si = 0; // 静态成员变量的初始化(Integral type)

const int BClass::csi = 1; // 静态常量成员变量的初始化(Integral type)

const double BClass::csd = 99.9; // 静态常量成员变量的初始化(non-Integral type)
// 在初始化(1)中的csi2时,根据著名大师Stanley B.Lippman的说法下面这行是必须的。

// 但在VC2003中如果有下面一行将会产生错误,而在VC2005中,下面这行则可有可无,这个和编译器有关。

const int BClass::csi2;
int main()

{

BClass b;

return 0;

}

---------------------------------------------------------------------------------------------
静态成员属于类作用域,但不属于类对象,和普通的static变量一样,程序一运行就分配内存并初始化,生命周期和程序一致。

所以,在类的构造函数里初始化static变量显然是不合理的。

静态成员其实和全局变量地位是一样的,只不过编译器把它的使用限制在类作用域内(不是类对象,它不属于类对象成员),要在类的定义外(不是类作用域外)初始化。

C++类中各种不同类型成员根据是否static 、是否const类型的初始化方法不尽相同,写代码时经常搞乱,网上搜了一下资料,全部总结一下。一、例子

[cpp] view
plaincopy

-----------------Test.h----------------------------

#pragma once

class Test

{

private :

int var1;

// int var11= 4; 错误的初始化方法

const int var2 ;

// const int var22 =22222; 错误的初始化方法,const成员和引用成员必须在初始化列表中初始化(因为它们必须在定义时初始化)

static int var3;

// static int var3333=33333; 错误,只有静态常量成员才能直接赋值来初始化

static const int var4=4444; //正确,静态常量整数成员(此三个条件缺一不可)可以直接初始化

static const int var44;

public:

Test(void);

~Test(void);

};

[cpp] view
plaincopy

----------Test.cpp----------------

#include ".\test.h"

int Test::var3 = 3333333; //静态成员的 正确的初始化方法

// int Test::var1 = 11111;; 错误 静态成员才能这样初始化

// int Test::var2 = 22222; 错误

// int Test::var44 = 44444; // 错误的方法,提示重定义,因为此var44与类中声明的var44的修饰符不相同

const int Test::var44 = 44444; //正确

Test::Test(void) :var1(11111),var2(22222) //正确的初始化方法 , var3(33333) 不能在这里初始化

{

var1 =11111; //正确, 普通变量也可以在这里初始化

//var2 = 222222; 错误,因为const常量不能赋值,只能在 构造函数的初始化列表 那里初始化

var3 =33; //经测试,(1)若已经正确定义了var3,即在类定义外部“int Test::var3;”或“int Test::var3=3333333;”,

//则此处赋值(注意是赋值,不是初始化)是正确的 ;

//(2)若按(1)所说的正确定义var3,则此处赋值是错误的,因为var3尚未定义。

}

Test::~Test(void)

二、简单概括

有些成员变量的数据类型比较特别,它们的初始化方式也和普通数据类型的成员变量有所不同。这些特殊的类型的成员变量包括:

a. 常量型成员变量:如 const int temp;

b. 引用型成员变量:如 int & temp;

c. 静态成员变量:如 static int temp;

d. 整型静态常量成员变量:如 static const int temp;

e. 非整型静态常量成员变量:如 static const double temp;

对于常量型成员变量和引用型成员变量的初始化,必须通过构造函数初始化列表的方式进行。在构造函数体内给常量型成员变量和引用型成员变量赋值的方式是行不通的。

静态成员变量的初始化也颇有点特别。

参考下面的代码以及其中注释:

[cpp] view
plaincopy

// Initialization of Special Data Member

#include <iostream>

using namespace std;

class BClass

{

private:

int i; // 普通成员变量

const int ci; // 常量成员变量

int &ri;

// 引用成员变量

static int si; // 静态成员变量

//static int si2 = 100; // error: 只有静态常量成员变量,才可以这样初始化

static const int csi; // 静态常量成员变量

static const int csi2 = 100; // 静态常量成员变量的初始化(Integral type) (1)

static const double csd; // 静态常量成员变量(non-Integral type)

//static const double csd2 = 99.9; // error: 只有静态常量整型数据成员才可以在类中初始化

public:

BClass() : i(1), ci(2), ri(i) // 对于常量型成员变量和引用型成员变量,必须通过

{ // 参数化列表的方式进行初始化。在构造函数体内进行赋值是不行的

}

};

// 静态成员变量的初始化(Integral type)

int BClass::si = 0;

// 静态常量成员变量的初始化(Integral type)

const int BClass::csi = 1;

// 静态常量成员变量的初始化(non-Integral type)

const double BClass::csd = 99.9;

// 在初始化(1)中的csi2时,根据Stanley B. Lippman的说法下面这行是必须的。

// 但在VC2003中如果有下面一行将会产生错误,而在VC2005中,下面这行则可有可无,这个和编译器有关。

const int BClass::csi2;

int main(void)

{

BClass b_class;

return 0;

}

三、完整总结

1、普通的变量:一般不考虑啥效率的情况下 可以在构造函数中进行赋值。考虑一下效率的可以再构造函数的初始化列表中进行。

[cpp] view
plaincopy

class CA

{

public:

int data;

……

public:

CA();

……

};

CA::CA():data(0)//……#1……初始化列表方式

{

//data = 0;//……#1……赋值方式

};

2、static 静态变量:

static变量属于类所有,而不属于类的对象,因此不管类被实例化了多少个对象,该变量都只有一个。

[cpp] view
plaincopy

class CA

{

public:

static int sum;

……

public:

CA();

……

};

int CA::sum=0;//……#……类外进行定义和初始化

3、const 常量变量:

const常量需要在声明的时候即初始化。因此需要在变量创建的时候进行初始化。一般采用在构造函数的初始化列表中进行。

[cpp] view
plaincopy

class CA

{

public:

const int max;

……

public:

CA();

……

};

CA::CA():max(100)

{

……

}

4、Reference 引用型变量:

引用型变量和const变量类似。需要在创建的时候即进行初始化。也是在初始化列表中进行。但需要注意用Reference类型。

[cpp] view
plaincopy

class CA

{

public:

int& counter;

……

public:

CA(int i);

……

};

CA::CA(int i):counter(i)

{

……

}

5、const static integral 变量:

对于既是const又是static 而且还是整形变量,C++是给予特权的。可以直接在类的定义中初始化。short可以,但float的不可以哦。

[cpp] view
plaincopy

class CA

{

public:

//static const float fmin = 0.0;

// only static const integral data members can be initialized within a class

const static int nmin = 0;

……

public:

……

};

总结起来,可以初始化的情况有如下四个地方:

1、在类的定义中进行的,只有const 且 static 且 integral 的变量。

2、在类的构造函数初始化列表中, 包括const对象和Reference对象。

3、在类的定义之外初始化的,包括static变量。因为它是属于类的唯一变量。

4、普通的变量可以在构造函数的内部,通过赋值方式进行。当然这样效率不高。

【资料来源:http://www.douban.com/note/66957147/http://blog.csdn.net/jenghau/article/details/4752735,有修改】
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: