C++ static、const和static const类型成员变量声明以及初始化

下面的文章是我从网上找了好久,整理了好久的,对我有很大帮助,原址已经记不住了,算我自己原创的了哇!

C++ staticconststatic const 以及它们的初始化

const定义的常量在超出其作用域之后其空间会被释放,而static定义的静态常量在函数执行后不会释放其存储空间

static表示的是静态的。类的静态成员函数、静态成员变量是和类相关的,而不是和类的具体对象相关的。即使没有具体对象,也能调用类的静态成员函数和成员变量。一般类的静态函数几乎就是一个全局函数,只不过它的作用域限于包含它的文件中。

在C++中,static静态成员变量不能在类的内部初始化。在类的内部只是声明,定义必须在类定义体的外部,通常在类的实现文件中初始化,如:double Account::Rate = 2.25;static关键字只能用于类定义体内部的声明中,定义时不能标示为static

在C++中,const成员变量也不能在类定义处初始化,只能通过构造函数初始化列表进行,并且必须有构造函数。

const数据成员 只在某个对象生存期内是常量,而对于整个类而言却是可变的因为类可以创建多个对象,不同的对象其const数据成员的值可以不同。所以不能在类的声明中初始化const数据成员,因为类的对象没被创建时,编译器不知道const数据成员的值是什么。

const数据成员的初始化只能在类的构造函数的初始化列表中进行。要想建立在整个类中都恒定的常量,应该用类中的枚举常量来实现,或者static cosnt。

class Test{
public:
	Test():a(0){}
	enum {size1=100,size2=200};
private:
	const int a;//只能在构造函数初始化列表中初始化
	static int b;//在类的实现文件中定义并初始化
	conststatic int c;//与 static const int c;相同。
};
 
int Test::b=0;//static成员变量不能在构造函数初始化列表中初始化,因为它不属于某个对象。
cosnt intTest::c=0;//注意:给静态成员变量赋值时,不需要加static修饰符,但要加cosnt。

cosnt成员函数主要目的是防止成员函数修改对象的内容。即const成员函数不能修改成员变量的值,但可以访问成员变量。当方法成员函数时,该函数只能是const成员函数。

static成员函数主要目的是作为类作用域的全局函数。不能访问类的非静态数据成员。类的静态成员函数没有this指针,这导致:1、不能直接存取类的非静态成员变量,调用非静态成员函数2、不能被声明为virtual

 

关于static、const、static cosnt、const static成员的初始化问题:

1、类里的const成员初始化:

在一个类里建立一个const时,不能给他初值

class foo{
public:
	foo():i(100){}
private:
	const int i=100;//error!!!
};
//或者通过这样的方式来进行初始化
foo::foo():i(100){}

 

2、类里的static成员初始化:

类中的static变量是属于类的,不属于某个对象,它在整个程序的运行过程中只有一个副本,因此不能在定义对象时 对变量进行初始化,就是不能用构造函数进行初始化,其正确的初始化方法是:

数据类型 类名::静态数据成员名=值;

class foo{
public:
	foo();
private:
staticint i;
};

intfoo::i=20;

.这表明:

1、初始化在类体外进行,而前面不加static,以免与一般静态变量或对象相混淆

2、初始化时不加该成员的访问权限控制符private、public等

3、初始化时使用作用域运算符来表明它所属的类,因此,静态数据成员是类的成员而不是对象的成员。

 

3、类里的static cosnt 和 const static成员初始化

这两种写法的作用一样,为了便于记忆,在此值说明一种通用的初始化方法:

class Test{
public:
	static const int mask1;
	conststatic int mask2;
};
constTest::mask1=0xffff;
constTest::mask2=0xffff;
//它们的初始化没有区别,虽然一个是静态常量一个是常量静态。静态都将存储在全局变量区域,其实最后结果都一样。可能在不同编译器内,不同处理,但最后结果都一样。

这是一个完整的例子:

#ifdef A_H_
#define A_H_
#include 
usingnamespace std;
class A{
public:
	A(int a);
	staticvoid print();//静态成员函数
private:
	static int aa;//静态数据成员的声明
	staticconst int count;//常量静态数据成员(可以在构造函数中初始化)
	const int bb;//常量数据成员
};

int A::aa=0;//静态成员的定义+初始化
const int A::count=25;//静态常量成员定义+初始化

A::A(int a):bb(a){//常量成员的初始化
aa+=1;
}

void A::print(){
cout<<"count="<#include …
class Myclass{
public:
	Myclass(int a, int b, int c);
	void GetNumber();
	void GetSum();
private:
	int A, B, C;
	static int Sum;
};

int Myclass::Sum = 0;

Myclass::Myclass(int a, int b, int c){
	A = a;
	B = b;
	C = c;
	Sum += A+B+C;
}

void Myclass::GetNumber(){
	cout<<"Number="<#include …
class M{
public:
	M(int a){
		A=a;
		B+=a;
	}
	
	static void f1(M m);
private:
	int A;
	static int B;
};

void M::f1(M m)
{
	cout<<"A="<-----------------Test.h----------------------------
#pragma once
class Test
{
private :
	int var1;
	// int var11= 4; 错误的初始化方法
	const int var2 ;
	// const int var22 =22222; 错误的初始化方法
	static int var3;
	// static int var3333=33333; 错误,只有静态常量int成员才能直接赋值来初始化
	static const int var4=4444; //正确,静态常量成员可以直接初始化	
	static const int var44;
public:
	Test(void);
	~Test(void);
};
--------------------Test.cpp-----------------------------------
#include ".\test.h"

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

// int Test::var1 = 11111;; 错误静态成员才能初始化
// int Test::var2 = 22222; 错误
// int Test::var44 = 44444; // 错误的方法,提示重定义
Test::Test(void):var1(11111),var2(22222)正确的初始化方法//var3(33333)不能在这里初始化
{
	var1 =11111; //正确, 普通变量也可以在这里初始化
	//var2 = 222222; 错误,因为常量不能赋值,只能在 “constructor initializer (构造函数的初始化列表)” 那里初始化
          
	var3 =44444; //这个赋值是正确的,不过因为所有对象一个静态成员,所以会影响到其他的,这不能叫做初始化了吧
}
Test::~Test(void){}
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

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

    a. 常量型成员变量
    b. 引用型成员变量
    c. 静态成员变量
    d. 整型静态常量成员变量
    e. 非整型静态常量成员变量

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

    静态成员变量的初始化也颇有点特别。
   参考下面的代码以及其中注释:

// Initialization of Special Data Member
#include 
using namespace std;     

class BClass
{
public:
	BClass() : i(1),ci(2), ri(i) // 对于常量型成员变量和引用型成员变量,必须通过	
	{				// 参数化列表的方式进行初始化。在构造函数体内进行赋值的方式,是行不通的。
	} 

	voidprint_values()
	{
		cout<< "i =\t" << i << endl;
		cout<< "ci =\t" << ci << endl;
		cout<< "ri =\t" << ri << endl;
		cout<< "si =\t" << si << endl;
		cout<< "csi =\t" << csi << endl;
		cout<< "csi2 =\t" << csi2 << endl;
		cout<< "csd =\t" << csd << endl;
	}

private:
	inti; // 普通成员变量
	constint ci; // 常量成员变量
	int&ri; // 引用成员变量
	staticint si; // 静态成员变量
	//staticint si2 = 100; // error: 只有静态常量成员变量,才可以这样初始化
	staticconst int csi; // 静态常量成员变量
	staticconst int csi2 = 100; // 静态常量成员变量的初始化(Integral type) (1)
	staticconst double csd; // 静态常量成员变量(non-Integral type)
	//staticconst double csd2 = 99.9; // error: 只有静态常量整型数据成员才可以在类中初始化
};
// 静态成员变量的初始化(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)
{
	BClassb_class;
	b_class.print_values(); 
	return0;
}
 

c++静态成员小结,c++,static

类中的静态成员真是个让人爱恨交加的特性。我曾经在面试时,被主考官抓住这个问题一阵穷追猛打,直把我问的面红耳赤,败下阵来。所以回来之后,我痛定思痛,决定好好总结一下静态类成员的知识点,以便自己在以后面试中,在此类问题上不在被动。 
静态类成员包括静态数据成员和静态函数成员两部分。 

静态数据成员

类体中的数据成员的声明前加上static关键字,该数据成员就成为了该类的静态数据成员。和其他数据成员一样,静态数据成员也遵守public/protected/private访问规则。同时,静态数据成员还具有以下特点: 

1.
静态数据成员的定义。 
静态数据成员实际上是类域中的全局变量。所以,静态数据成员的定义(初始化)不应该被放在头文件中。 声明vs.定义(初始化)
其定义方式与全局变量相同。举例如下: 

xxx.h文件 
class base{ 
private: 
    static const int _i;//声明,标准c++支持有序类型在类体中初始化,但vc6不支持。 
}; 

xxx.cpp文件 
const int base::_i=10;//定义(初始化)时不受private和protected访问限制. 


注:不要试图在头文件中定义(初始化)静态数据成员。在大多数的情况下,这样做会引起重复定义这样的错误。 即使加上 #ifndef #define #endif 或者 #pragma once 也不行。  

2.
静态数据成员被的所有对象所共享,包括该类派生类的对象。即派生类对象与基类对象共享基类的静态数据成员。 举例如下:  
class base{ 
public : 
	static int _num;//声明 
}; 
int base::_num=0;//静态数据成员的真正定义 

class derived:public base{ }; 

main() 
{ 
	base a; 
	derived b; 
	a._num++; 
	cout<<"base class static data number_num is"<



3. 静态数据成员可以成为成员函数的 可选参数,而普通数据成员则不可以。举例如下:  
class base{ 
public : 
	static int _staticVar; 
	int _var; 
	void foo1(int i=_staticVar);//正确,_staticVar为静态数据成员 
	void foo2(int i=_var);//错误,_var为普通数据成员 
}; 


4. 静态数据成员的类型可以是所属类的类型,而普通数据成员则不可以。普通数据成员的只能声明为所属类类型的指针或引用。 举例如下:  
class base{ 
public : 
	static base_object1;//正确,静态数据成员 
	base _object2;//错误 
	base *pObject;//正确,指针 
	base &mObject;//正确,引用 
}; 

5. 这个特性,我不知道是属于标准 c++ 中的特性,还是 vc6 自己的特性。  
静态数据成员的值在const成员函数中可以被合法的改变 。举例如下:  
class base{ 
public: 
	base(){
		_i=0;
		_val=0;
	} 

	mutable int _i; 
	static int _staticVal; 
	int _val; 
	void test() const{//const 成员函数 
		_i++;//正确,mutable数据成员 
		_staticVal++;//正确,static数据成员 
		_val++;//错误 
	} 
}; 
int base::_staticVal=0; 

 

静态成员函数 

静态成员函数没有什么太多好讲的。 

1.
静态成员函数的地址可用普通函数指针储存,而普通成员函数地址需要用类成员函数指针来储存。举例如下: 

class base{ 
    static int func1(); 
    int func2(); 
}; 

int (*pf1)()=&base::func1;//普通的函数指针 
int (base::*pf2)()=&base::func2;//成员函数指针 

2. 静态成员函数不可以调用类的非静态成员 。因为静态成员函数不含 this指针  

3.
静态成员函数不可以同时声明为 virtualconstvolatile函数 。举例如下:  
class base{ 
    virtual static void func1();//错误 
    static void func2() const;//错误 
    static void func3() volatile;//错误 
}; 


最后要说的一点是,静态成员是可以独立访问的,也就是说,无须创建任何对象实例就可以访问。

你可能感兴趣的:(C/C++,c++,static,const,成员变量,初始化)