C++ static详解

滚雪球~ 2024-07-03 12:05:01 阅读 72

参考:https://blog.csdn.net/u010797208/article/details/41549461

static详解

static 作为关键字出现在C++语言当中,其运用比较复杂,其修饰全局变量,局部变量,类成员变量,类成员函数等都有不同释义,下面为大家详细描述一下.

一 static概述

static 从笼统意义的理解上,可以理解为 被此关键字修饰过的函数或变量,即为静态函数或静态变量.

静态变量或函数式在系统编译期,main函数运行前就已经分配内存的.在声明它的程序块,子程序块或函数内部有效,值保持

虽然上面讲了static关键字修饰不同的类型单位会有不同的效果,但其实可以理解为面向过程程序中的static和面向对象程序中的static,

面向过程中的static主要适用于普通变量和函数.

面向对象中的static主要适用于类中的成员变量和成员函数.

二 static用法详解

面向过程中的static

静态全局变量

在全局变量之前,加上关键字static,就可以定义该变量为静态全局变量.

例:

<code>#include<stdio.h>

static int Temp = 10;

void Test();

int main()

{

printf("Temp is %d\n", Temp);//输出:10

Test();//输出:10

printf("Temp is %d\n", Temp);//输出:11

return 0;

}

void Test()

{

Temp++;

}

局部变量之前加上static,可以指定该局部变量为静态局部变量

代码中主要有一下几个特点

01.在全局数据中的变量如果没有显示的初始化会自动被程序初始化为0(这个特性非静态全局变量也有),而在函数体内声明的变量如果不显示初始化则会使一个随机值.

02.静态全局变量在声明它的整个文件中都是课件的,而在文件之外是不可见的,这点区别于全局变量.

03.静态全局变量在全局数据区分配内存

04.其他文件中可以定义同名int型变量Temp,不会冲突,各取各的值.

有一定C++语言功底的朋友可能看出,上面的示例代码 其实如果将Temp的static去掉,程序也可以正常执行.但是请注意以下区别.

01.全局变量时不显示调用static关键字修饰的变量,全局变量默认是有外部连接性的,其作用域是整个工程,在一个文件内定义的全局变量可以通过,包含其所在头文件或显示调用 extern关键字修饰全局变量的变量名声明来引用.

02.静态全局变量是显示调用static修饰的全局变量,其作用域只在声明此变量的文件中,其他文件即使使用extern关键字修饰其声明也不可使用.

03.关于第二点,其描述是C++标准,但是有些编译器会不执行次标准特意对其进行优化,目前亲测的vs2012与g++都不支持这个标准…个人表示也是醉了

静态局部变量

在局部变量之前加上static,可以指定该局部变量为静态局部变量

#include <stdio.h>

void Test();

int main()

{

for(int i=0; i<5; i++)

{

Test();

}

return 0;

}

void Test()

{

static int Temp = 10;

printf("Temp is %d\n", Temp);

Temp++;

}

以上程序输出为:

10

11

12

13

14

通常,在一个函数作用域内定义一个变量,每次运行到该函数时,系统会给局部变量分配内存,当函数结束时,该变量的内存会被系统回收至栈内存当中,局部变量也会消散.

但是如果改局部变量被声明为static 静态局部变量时,则该变量的生命周期不受到该函数的作用域限制,也就是说即使函数运行结束,系统仍然会保留该静态变量的内存,不会回收,它始终驻留在全局数据区当中,直到整个进程程序运行结束时,其内存才会被回收.

静态局部变量有以下特点:

01.其内存存放在 程序的全局数据区中,

02.静态局部变量在程序执行到该对象声明时,会被首次初始化.其后运行到该对象的声明时,不会再次初始化,这也是为什么上面程序测试函数每次输出的值都是递增的原因.

03.如果静态局部变量没有被显示初始化,则其值会自动被系统初始化为0.

04.局部静态变量 不能被其作用域之外的其他模块调用,其调用范围仅限于声明该变量的函数作用域当中.

静态函数

在函数的返回类型前加上关键字static,可以将此函数声明为静态函数.静态函数与普通函数不同,其作用域只在声明它的文件当中.其他文件可以定义同名的全局函数,而不冲突.

想要在其他文件调用静态函数 需要显示的调用extern关键字修饰其声明.否则编译器会link error.

#include<stdio.h>

static void Test();

int main()

{

Test();

return 0;

}

void Test()

{

printf("Is Static Fun\n");

}

总结定义静态函数的好处:

01.其他文件可以定义同名函数

02.静态函数不会被其他文件所引用,其作用域只在当前声明他的文件中.

面向对象中的static

静态数据成员

在类内的数据成员声明前加上关键字static,则该成员将会被声明为静态数据成员.

#include <stdio.h>

class TempClass

{

public:

TempClass(int a, int b, int c);

void Show();

private:

int a,b,c;

static int T;

};

int TempClass::T = 0;//初始化静态数据成员

TempClass::TempClass(int a, int b, int c)

{

this->a = a;

this->b = b;

this->c = c;

T = a + b + c;

}

void TempClass::Show()

{

printf("T is %d\n", T);

}

int main()

{

TempClass ClassA(1,1,1);

ClassA.Show();//输出1+1+1 = 3;

TempClass ClassB(3,3,3);

ClassB.Show();//输出3+3+3 = 9;

ClassA.Show();//输出9

return 0;

}

从上面的测试代码可以看出 静态数据成员的特点:

01.静态数据成员的服务对象并非是单个类实例化的对象,而是所有类实例化的对象(这点可以用于设计模式中的单例模式实现).

02.静态数据成员必须显示的初始化分配内存,在其包含类没有任何实例花之前,其已经有内存分配.

03.静态数据成员与其他成员一样,遵从public,protected,private的访问规则.

04.静态数据成员内存存储在全局数据区,只随着进程的消亡而消亡.

静态数据成员与全局变量相比的优势:

01.静态数据成员不进入程序全局名字空间,不会与其他全局名称的同名同类型变量冲突.

02.静态数据成员可以实现C++的封装特性,由于其遵守类的访问权限规则.所以相比全局变量更加灵活.

静态成员函数

在类的成员函数返回类型之前添加static,即可声明此成员函数为静态成员函数.

#include <stdio.h>

class TempClass

{

public:

TempClass(int a, int b, int c);

static void Show();

private:

int a,b,c;

static int T;

};

int TempClass::T = 0;//初始化静态数据成员

TempClass::TempClass(int a, int b, int c)

{

this->a = a;

this->b = b;

this->c = c;

T = a + b + c;

}

void TempClass::Show()

{

printf("T is %d\n", T);

}

int main()

{

TempClass ClassA(1,1,1);

ClassA.Show();

TempClass ClassB(3,3,3);

ClassB.Show();

TempClass::Show();//注意此处的调用方式.

return 0;

}

从上面的示例代码中可以看出 静态局部函数的特点如下:

01.静态成员函数比普通成员函数多了一种调用方式.

02.静态成员函数为整个类服务,而不是具体的一个类的实例服务.(这句话可能比较绕口,可以理解为在没有任何实例化的类对象的条件下调用类方法,详见上面代码注释处.)

03.静态成员函数中没有隐含的this指针,所以静态成员函数不可以操作类中的非静态成员.

ps:关于this指针的深入解释

在C++中,普通的成员函数一般都隐含了一个this指针,例如调用函数Fun(),实际上是this->Fun().

静态成员函数中没有这样的this指针,所以静态成员函数不能操作类中的非静态成员函数.否则编译器会报错.

三 注意事项

01.静态数据成员都是静态存储的,所以必须在main函数之前显示的对其进行初始化.

02.静态成员初始化与一般成员的初始化不同.

03.不能再头文件中声明静态全局变量,这点在简单的测试代码中无法体现,只有在多文件同时包含,引用和操作时候才会显露出来.其结果可能是产生了多个同名的静态数据.一旦出现这种问题,是非常难以查找和排除的.

04.不能将静态成员函数定义为虚函数.

05.静态成员函数没有this指针.

06.static缩短了子类对父类静态成员访问的时间,相对来说节省了内存空间

07.关于06条的补充,如果不想在子类中操作父类的静态成员,则可以在子类中定义一个同名的static成员.这样既可覆盖父类中的静态成员.并且根据C++的多态性变量命名规则.这样做是安全的.

08.静态成员声明在类中,操作在其外部,所以对其取地址操作就跟取普通成员的操作略有不同.静态变量地址是指向其数据类型的指针,函数地址则是一个类型为nonmember的函数指针.



声明

本文内容仅代表作者观点,或转载于其他网站,本站不以此文作为商业用途
如有涉及侵权,请联系本站进行删除
转载本站原创文章,请注明来源及作者。