C/C++中static再探讨

1. C 语言中的static

内存布局:

初始化数据段---.data段
通常将此段称为数据段,它包含了程序中需赋初值的变量。初始化 的全局变量和静态变量存放在这里 。例如,C程序中任何函数之外的说明:int maxcount = 99; 使此变量以初值存放在初始化数据段中。
a.初始化的全局变量
b.初始化的静态变量
非初始化数据段---.bss段
通常将此段称为bss段,这一名称来源于早期汇编程序的一个操作符,意思是“block started by symbol(由符号开始的块)”,未初始化 的全局变量 和静态变量存放在这里。在程序开始执行之前,内核将此段初始化为0。函数外的说明:long sum[1000] ; 使此变量存放在非初始化数据段中。
a.未初始化的全局变量
b.未初始化的静态变量
 

static 局部变量

静态局部变量属于静态存储方式,它具有以下特点:
(1)静态局部变量 在函数内定义它的生存期为 整个程序生命周期,但是其作用域仍与自动变量相同 ,只能在定义该变量的函数内使用该变量。退出该函数后,尽管该变量还继续存在,但不能使用它。
(2)对基本类型的静态局部变量若在声明时未赋以初值,则系统自动赋予0值 。而对自动变量不赋初值,则其值是不定的。
根据静态局部变量的特点,可以看出它是一种生存期为整个程序生命周期。虽然离开定义它的函数后不能使用,但如再次调用定义它的函数时,它又可继续使用, 而且保存了前次被调用后留下的值。 因此,当多次调用一个函数且要求在调用之间保留某些变量的值时,可考虑采用静态局部变量 。虽然用全局变量也可以达到上述目的,但全局变量有时会造成意外的副作用,因此仍以采用局部静态变量为宜。
 

static int j;
void fun1()
{
    static int i = 0;
    i++;
}
void fun2()
{
    j = 0;
    j++;
}
void fun3()
{
    int k = 0;
    k++;
}
int main()
{
    for(int k= 0; k < 10; k++)
    {
        fun1();
        fun2();
        fun3();
    }
    return 0;
}

全局静态变量j在.bss段(未初始化),全局静态变量i在.bss段(初始化为0),局部变量k在栈上。fun1中将局部变量改变为静态之后初始化一次,后面每次进入循环之后直接进行++操作,变量i的值不随函数栈帧的释放而释放;fun2中,对全局变量j的值先进行0赋值再+操作;fun3中k随栈帧开辟而定义,随栈帧消退而释放 。
从以上分析可以看出, 把局部变量改变为静态变量后是改变了它的存储方式即改变了它的生存期 。

static 全局变量

全局变量(外部变量)的说明之前再冠以static 就构成了静态的全局变量。全局变量本身就是静态存储方式, 静态全局变量当然也是静态存储方式。 这两者在存储方式上并无不同。
这两者的区别在于:
(1). 非静态全局变量 的作用域是整个源程序 ,当一个源程序由多个源文件组成时,非静态的全局变量在各个源文件中都是有效的。
(2). 而静态全局变量 则限制了其作用域,即只在定义该变量的源文件内有效,在同一源程序的其它源文件中不能使用它。由于静态全局变量的作用域局限于 一个源文件内 ,只能为该源文件内的函数公用, 因此可以避免在其它源文件中引起错误。
从以上分析可以看出,把全局变量改变为静态变量后是改变了它的作用域, 限制了它的使用范围 。

static 函数

如果在一个源文件中定义的函数,只能被本文件中的函数调用,而不能被同一程序其它文件中的函数调用,这种函数称为static函数与称为静态函数。 
定义一个static函数,只需在函数类型前再加一个“static”关键字即可,如下所示:
static 函数类型 函数名(函数参数表) {……}
关键字“static”,译成中文就是“静态的”,所以内部函数又称静态函数。但此处“static”的含义不是指存储方式,而是指 对函数的作用域仅局限于本文件 。
使用内部函数的好处是:不同的人编写不同的函数时,不用担心自己定义的函数,是否会与其它文件中的函数同名,因为同名也没有关系 。

2. C++中的static

static 数据成员

在类内数据成员的声明前加上关键字static,该数据成员就是类内的静态数据成员。先举一个静态数据成员的例子。

#include <iostream.h>
class Myclass
{
public:
    Myclass(int a,int b,int c);
    void GetSum();
private:
    int a,b,c;
    static int Sum; //声明静态数据成员
};
int Myclass::Sum=0; //定义并初始化静态数据成员
//static int Myclass::Sum = 0; //注意加static, 是错误的
Myclass::Myclass(int a,int b,int c)
{
    this->a=a;
    this->b=b;
    this->c=c;
    Sum+=a+b+c;
}
void Myclass::GetSum()
{
    cout<<"Sum="<<Sum<<endl;
}
int main()
{
    Myclass M(1,2,3);
    M.GetSum(); // cout 6
    Myclass N(4,5,6); 
    N.GetSum(); // cout 21
    M.GetSum(); // cout 21
    return 0;
}


可以看出,static数据成员有以下特点:
(1). 对于非static数据成员,每个类对象都有自己的拷贝。而static数据成员被当作是类的成员。无论这个类的对象被定义了多少个,静态数据成员在程序中也只有一份拷贝 ,由该类型的所有对象共享访问。也就是说,静态数据成员是该类的所有对象所共有的。对该类的多个对象来说,静态数据成员只分配一次内存,供所有对象共用。
(2). 静态数据成员存储在全局数据区。静态数据成员定义时才分配空间,所以不能在类声明中定义 。在上例中,语句int Myclass::Sum = 0;是定义静态数据成员;
(3). 静态数据成员和普通数据成员一样遵从public,protected,private访问规则 ;
(4). 因为静态数据成员在全局数据区分配内存,属于本类的所有对象共享,所以,它不属于特定的类对象,在没有产生类对象时其作用域就可见,即在没有产生类的实例时,我们就可以操作它;
(5). 静态数据成员初始化与一般数据成员初始化不同。静态数据成员初始化的格式为:<数据类型><类名>::<静态数据成员名>=<值> 如:int Myclass::Sum=0;
(6). 类的静态数据成员有两种访问形式:<类对象名>.<静态数据成员名> 或 <类类型名>::<静态数据成员名>
M.Sum = 0 或 Myclass::Sum = 0 (但是上面这个例子是不行的,因为他是private的变量)如果静态数据成员的访问权限允许的话(即public的成员),可在程序中,按上述格式来引用静态数据成员 ;
(7). 静态数据成员主要用在各个对象都有相同的某项属性的时候。比如对于一个存款类,每个实例的利息都是相同的。所以,应该把利息设为存款类的静态数据成员。这有两个好处,第一,不管定义多少个存款类对象,利息数据成员都共享分配在全局数据区的内存,所以节省存储空间。第二,一旦利息需要改变时,只要改变一次,则所有存款类对象的利息全改变过来了;
(8). 同全局变量相比,使用静态数据成员有两个优势:
a. 静态数据成员没有进入程序的全局名字空间,因此不存在与程序中其它全局名字冲突的可能性;
b. 可以实现信息隐藏。静态数据成员可以是private成员,而全局变量不能;

static 成员函数

static 成员函数,它为类的全部服务而不是为某一个类的具体对象服务。普通的成员函数一般都隐含了一个this指针,因为普通成员函数总是具体的属于某个类的具体对象的。通常情况下,this是缺省的。如函数fn()实际上是this->fn()。但是与普通函数相比,静态成员函数由于不是与任何的对象相联系,因此它不具有this指针 。从这个意义上讲,它无法访问属于类对象的no-static数据成员,也无法访问no-static成员函数,它只能调用其余的静态成员函数 。下面举个静态成员函数的例子。

#include <iostream.h>
using namespace std;
class Myclass
{
public:
    Myclass(int a,int b,int c);
    static void GetSum(); /声明静态成员函数
private:
    int a,b,c;
    static int Sum; //声明静态数据成员
};
int Myclass::Sum = 0;//定义并初始化静态数据成员
Myclass::Myclass(int a,int b,int c)
{
    this->a=a;
    this->b=b;
    this->c=c;
    Sum+=a+b+c; //非静态成员函数可以访问静态数据成员
}
    //static void Myclass::GetSum(){...} //加上static是错误的
    void Myclass::GetSum() //静态成员函数的实现
{
//cout<<a<<endl; //错误代码,a是非静态数据成员
cout<<"Sum="<<Sum<<endl; //静态函数是能访问静态数据成员
}
int main()
{
    Myclass M(1,2,3);
    M.GetSum();
    Myclass N(4,5,6);
    N.GetSum();
    Myclass::GetSum();
    return 0;
}

关于静态成员函数,可以总结为以下几点:
(1). 出现在类体外的函数定义不能指定关键字static ;
(2). static成员之间可以相互访问 ,包括static成员函数访问static数据成员和访问static成员函数;
(3). 非静态成员函数可以任意地访问静态成员函数和静态数据成员;
(4). 静态成员函数不能访问非静态成员函数和非静态数据成员,只能访问静态的 ;
(5). 由于没有this指针的额外开销,因此静态成员函数与类的全局函数相比速度上会有少许的增长;
(6). 调用静态成员函数,可以用成员访问操作符(.)和(->)为一个类的对象或指向类对象的指针调用静态成员函数,也可以直接使用如下格式:
<类名>::<静态成员函数名>(<参数表>)
如:Myclass::GetSum(),调用类的静态成员函数。
但是,一样要遵从public,protected,private访问规则 。

猜你喜欢

转载自blog.csdn.net/ox0080/article/details/83686968