C++关键字static解析总结

http://developer.51cto.com/art/201104/254141.htm(下文引用地址)

要理解static,就必须要先理解另一个与之相对的关键字,很多人可能都还不知道有这个关键字,那就是auto,其实我们通常声明的不用static修饰的变量,都是auto的,因为它是默认的,就象short和long总是默认为int一样;我们通常声明一个变量:

int a;

string s;

其实就是:

auto int a;

auto string s;

而static变量的声明是:

static int a;

static string s;

这样似乎可以更有利于理解auto和static是一对成对的关键字吧,就像private,protected,public一样;

对于static的不理解,其实就是对于auto的不理解,因为它是更一般的;有的东西你天天在用,但未必就代表你真正了解它;auto的含义是由程序自动控制变量的生存周期,通常指的就是变量在进入其作用域的时候被分配,离开其作用域的时候被释放;而static就是不auto,变量在程序初始化时被分配,直到程序退出前才被释放;也就是static是按照程序的生命周期来分配释放变量的,而不是变量自己的生命周期;所以,像这样的例子:

  1. void func()
  2. {
  3. int a;
  4. static int b;
  5. }

每一次调用该函数,变量a都是新的,因为它是在进入函数体的时候被分配,退出函数体的时候被释放,所以多个线程调用该函数,都会拥有各自独立的变量a,因为它总是要被重新分配的;而变量b不管你是否使用该函数,在程序初始化时就被分配的了,或者在第一次执行到它的声明的时候分配(不同的编译器可能不同),所以多个线程调用该函数的时候,总是访问同一个变量b,这也是在多线程编程中必须注意的!

http://blog.csdn.net/wenrenhua08/article/details/39990485(下文引用地址)

static从宏观上讲主要有两种用法,一、是面向过程设计;二是面向对象设计。前主要是涉及变量与函数的用法;后者呢包含前者用法的同时,还有在类中的用法。

一、 面向过程设计中的static(C语言)

在讲面向过程设计中的static用法时先搞点插曲,从历史上讲,C程序一直由下面几部分组成:

正文段

CPU执行的机器指令部分。通常,正文段是可共享的,所以即使是经常环境指针环境表环境字符串执行的程序(如文本编辑程序、C编译程序、s h e l l等)在存储器中也只需有一个副本,另外,正文段常常是只读的,以防止程序由于意外事故而修改其自身的指令。

初始化数据段

通常将此段称为数据段,它包含了程序中需赋初值的变量。初始化 的全局变量和 静态变量存放在这里。例如,C程序中任何函数之外的说明:int maxcount = 99; 使此变量以初值存放在初始化数据段中。

a.初始化的全局变量

b.初始化的静态变量

非初始化数据段

通常将此段称为bss段,这一名称来源于早期汇编程序的一个操作符,意思是“block started by symbol(由符号开始的块)”,未初始化的全局变量 和静态变量存放在这里。在程序开始执行之前,内核将此段初始化为0。函数外的说明:long sum[1000] ; 使此变量存放在非初始化数据段中。

a.未初始化的全局变量

b.未初始化的静态变量

需要由程序员分配释放管理,若程序员不释放,程序结束时可能由OS回收。通常在堆中进行动态存储分配。如程序中的malloc, calloc,realloc等函数都从这里面分配。堆是从下向上分配的。

由编译器自动分配释放管 理。局部变量及每次函数调用时返回地址、以及调用者的环境信息(例如某些机器寄存器)都存放在栈中。新被调用的函数在栈上为其自动和临时变量分配存储空间。通过以这种方式使用栈,C函数可以递归调用。递归函数每次调用自身时,就使用一个新的栈帧,因此一个函数调用实例中的变量绝不会影响另一个函数调用实例中的变量。

a.局部变量

b.函数调用时返回地址

c.调用者的环境信息(例如某些机器寄存器)

在这我先把static的内部机制与优势先提到前面来讲述,本来想放到面向对象设计中来讲,但是它的重要性改变了初衷:

static的内部机制:

静态数据成员要在程序一开始运行时就必须存在。因为函数在程序运行中被调用,所以静态数据成员不能在任何函数内分配空间和初始化。

这样,它的空间分配有三个可能的地方,一是作为类的外部接口的头文件,那里有类声明;二是类定义的内部实现,那里有类的成员函数定义;三是应用程序的main()函数前的全局数据声明和定义处。

静态数据成员要实际地分配空间,故不能在类的声明中定义(只能声明数据成员)。类声明只声明一个类的“尺寸和规格”,并不进行实际的内存分配,所以在类声明中写成定义是错误的。它也不能在头文件中类声明的外部定义,因为那会造成在多个使用该类的源文件中,对其重复定义。

static被引入以告知编译器,将变量存储在程序的静态存储区而非栈上空间,静态数据成员按定义出现的先后顺序依次初始化,注意静态成员嵌套时,要保证所嵌套的成员已经初始化了。消除时的顺序是初始化的反顺序。

static的优势:

可以节省内存,因为它是所有对象所公有的,因此,对多个对象来说,静态数据成员只存储一处,供所有对象共用。静态数据成员的值对每个对象都是一样,但它的值是可以更新的。只要对静态数据成员的值更新一次,保证所有对象存取更新后的相同的值,这样可以提高时间效率。

一、static局部变量

静态局部变量属于静态存储方式,它具有以下特点:

(1)静态局部变量 在函数内定义它的生存期为 整个程序生命周期,但是其 作用域仍与 自动变量相同 ,只能在定义该变量的函数内使用该变量。退出该函数后,尽管该变量还继续存在,但不能使用它。

(2)对基本类型的静态局部变量若在声明时未赋以初值,则系统自动赋予0值 。而对自动变量不赋初值,则其值是不定的。

根据静态局部变量的特点,可以看出它是一种生存期为整个程序生命周期。虽然离开定义它的函数后不能使用,但如再次调用定义它的函数时,它又可继续使用,而且保存了前次被调用后留下的值。因此,当多次调用一个函数且要求在调用之间保留某些变量的值时,可考虑采用静态局部变量。虽然用全局变量也可以达到上述目的,但全局变量有时会造成意外的副作用,因此仍以采用局部静态变量为宜。

[cpp] view
plain
copyprint?

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. void Test()
  4. {
  5. static int tmpValue = 0;
  6. printf("value = %d\n", ++tmpValue);
  7. }
  8. int main()
  9. {
  10. for(int index = 0; index < 50; ++index)
  11. {
  12. Test()
  13. }
  14. getchar();
  15. return 0;
  16. }

二、全局变量

全局变量(外部变量)的说明之前再冠以static 就构成了静态的全局变量。全局变量本身就是静态存储方式,静态全局变量当然也是静态存储方式。 这两者在存储方式上并无不同。

这两者的区别在于:

(1). 非静态全局变量 的作用域是整个源程序 ,当一个源程序由多个源文件组成时,非静态的全局变量在各个源文件中都是有效的。

(2). 而静态全局变量 则限制了其作用域,即只在定义该变量的源文件内有效,在同一源程序的其它源文件中不能使用它。由于静态全局变量的作用域局限于 一个源文件内 ,只能为该源文件内的函数公用,
因此可以避免在其它源文件中引起错误。

从以上分析可以看出,把全局变量改变为静态变量后是改变了它的作用域,限制了它的使用范围 。

三、static 函数

如果在一个源文件中定义的函数,只能被本文件中的函数调用,而不能被同一程序其它文件中的函数调用,这种函数称为static函数与称为静态函数。

定义一个static函数,只需在函数类型前再加一个“static”关键字即可,如下所示:

static 函数类型 函数名(函数参数表) {……}

关键字“static”,译成中文就是“静态的”,所以内部函数又称静态函数。但此处“static”的含义不是指存储方式,而是指对函数的作用域仅局限于本文件 。

使用内部函数的好处是:不同的人编写不同的函数时,不用担心自己定义的函数,是否会与其它文件中的函数同名,因为同名也没有关系。

二、面向对象设计中的static(C++语言)

一、static 数据成员

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

[cpp] view
plain
copyprint?

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. class Test
  4. {
  5. public:
  6. Test() {}
  7. ~Test() {}
  8. Test(const Test& t)
  9. {
  10. if(this != &t)
  11. {
  12. this->m_nTest1 = t.m_nTest1;
  13. this->m_nTest2 = t.m_nTest2;
  14. this->m_nTest3 = t.m_nTest3;
  15. }
  16. }
  17. Test& operator=(const Test& t)
  18. {
  19. if(this != &t)
  20. {
  21. this->m_nTest1 = t.m_nTest1;
  22. this->m_nTest2 = t.m_nTest2;
  23. this->m_nTest3 = t.m_nTest3;
  24. }
  25. return *this;
  26. }
  27. void PrintOut()
  28. {
  29. //printf("test_1 = %d test_2 = %d test_3 = %d\n",m_nSTest1, m_nSTest2, m_nSTest3);//错误
  30. printf("test_1 = %d test_2 = %d test_3 = %d\n",m_nTest1, m_nTest2, m_nTest3);
  31. }
  32. static void PrintStatic()
  33. {
  34. printf("test_1 = %d test_2 = %d test_3 = %d\n",m_nSTest1, m_nSTest2, m_nSTest3);
  35. //printf("test_1 = %d test_2 = %d test_3 = %d\n",m_nTest1, m_nTest2, m_nTest3);//错误
  36. }
  37. public:
  38. int m_nTest1;
  39. static int m_nSTest1;
  40. protect:
  41. int m_nTest2;
  42. static int m_nSTest2;
  43. private:
  44. int m_nTest3;
  45. static int m_nSTest3;
  46. };
  47. int Test::m_nSTest1 = 10;
  48. int Test::m_nSTest2 = 20;
  49. int Test::m_nSTest3 = 30;
  50. int Test::m_nSTest1 = 10;
  51. int main()
  52. {
  53. Test t;
  54. t.PrintOut();
  55. //t.PrintStatic();//错误
  56. Test::PrintStatic();
  57. getchar();
  58. return 0;
  59. }

static数据成员有以下特点:

(1). 对于非static数据成员,每个类对象都有自己的拷贝。而static数据成员被当作是类的成员。无论这个类的对象被定义了多少个,静态数据成员在程序中也只有一份拷贝 ,由该类型的所有对象共享访问。也就是说,静态数据成员是该类的所有对象所共有的。对该类的多个对象来说,静态数据成员只分配一次内存,供所有对象共用。

(2). 静态数据成员存储在全局数据区。静态数据成员定义时才分配空间,所以不能在类声明中定义。在上例中,语句int Test::m_nSTest1= 10;是定义静态数据成员;

(3). 静态数据成员和普通数据成员一样遵从public,protected,private访问规则;

(4). 因为静态数据成员在全局数据区分配内存,属于本类的所有对象共享,所以,它不属于特定的类对象,在没有产生类对象时其作用域就可见,即在没有产生类的实例时,我们就可以操作它;

(5). 静态数据成员初始化与一般数据成员初始化不同。静态数据成员初始化的格式为:<数据类型><类名>::<静态数据成员名>=<值> 如:int Test::m_nSTest1 = 10;

(6). 类的静态数据成员有两种访问形式:<类对象名>.<静态数据成员名> 或 <类类型名>::<静态数据成员名>

(7). 静态数据成员主要用在各个对象都有相同的某项属性的时候。比如对于一个存款类,每个实例的利息都是相同的。所以,应该把利息设为存款类的静态数据成员。这有两个好处,第一,不管定义多少个存款类对象,利息数据成员都共享分配在全局数据区的内存,所以节省存储空间。第二,一旦利息需要改变时,只要改变一次,则所有存款类对象的利息全改变过来了;

(8). 同全局变量相比,使用静态数据成员有两个优势:

a. 静态数据成员没有进入程序的全局名字空间,因此不存在与程序中其它全局名字冲突的可能性;

b. 可以实现信息隐藏。静态数据成员可以是private成员,而全局变量不能;

二、static成员函数

[cpp] view
plain
copyprint?

  1. #include <stdio.h>
  2. #include <stdlib.h>
  3. class Test
  4. {
  5. public:
  6. Test() {}
  7. ~Test() {}
  8. Test(const Test& t)
  9. {
  10. if(this != &t)
  11. {
  12. this->m_nTest1 = t.m_nTest1;
  13. this->m_nTest2 = t.m_nTest2;
  14. this->m_nTest3 = t.m_nTest3;
  15. }
  16. }
  17. Test& operator=(const Test& t)
  18. {
  19. if(this != &t)
  20. {
  21. this->m_nTest1 = t.m_nTest1;
  22. this->m_nTest2 = t.m_nTest2;
  23. this->m_nTest3 = t.m_nTest3;
  24. }
  25. return *this;
  26. }
  27. void PrintOut()
  28. {
  29. //printf("test_1 = %d test_2 = %d test_3 = %d\n",m_nSTest1, m_nSTest2, m_nSTest3);//错误
  30. printf("test_1 = %d test_2 = %d test_3 = %d\n",m_nTest1, m_nTest2, m_nTest3);
  31. }
  32. static void PrintStatic()
  33. {
  34. printf("test_1 = %d test_2 = %d test_3 = %d\n",m_nSTest1, m_nSTest2, m_nSTest3);
  35. //printf("test_1 = %d test_2 = %d test_3 = %d\n",m_nTest1, m_nTest2, m_nTest3);//错误
  36. }
  37. public:
  38. int m_nTest1;
  39. static int m_nSTest1;
  40. protect:
  41. int m_nTest2;
  42. static int m_nSTest2;
  43. private:
  44. int m_nTest3;
  45. static int m_nSTest3;
  46. };
  47. int Test::m_nSTest1 = 10;
  48. int Test::m_nSTest2 = 20;
  49. int Test::m_nSTest3 = 30;
  50. int Test::m_nSTest1 = 10;
  51. int main()
  52. {
  53. Test t;
  54. t.PrintOut();
  55. //t.PrintStatic();//错误
  56. Test::PrintStatic();
  57. getchar();
  58. return 0;
  59. }

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

关于静态成员函数,可以总结为以下几点:

(1). 出现在类体外的函数定义不能指定关键字static ;

(2). static成员之间可以相互访问 ,包括static成员函数访问static数据成员和访问static成员函数;

(3). 非静态成员函数可以任意地访问静态成员函数和静态数据成员;

(4). 静态成员函数不能访问非静态成员函数和非静态数据成员,只能访问静态的;

(5). 由于没有this指针的额外开销,因此静态成员函数与类的全局函数相比速度上会有少许的增长;

(6). 调用静态成员函数,可以用成员访问操作符(.)和(->)为一个类的对象或指向类对象的指针调用静态成员函数,也可以直接使用如下格式:

<类名>::<静态成员函数名>(<参数表>)

如:Test::PrintStatic(),调用类的静态成员函数。

但是,一样要遵从public,protected,private访问规则。

时间: 2024-10-04 21:39:20

C++关键字static解析总结的相关文章

关键字static完全解析

在c语言中关键字static的意思是静态,它有三个明显的作用. 1).在函数体内,静态变量具有"记忆功能",即一个变量被声明为静态变量,在函数下次被调用时仍保留上次调用该函数时该静态变量的值. 代码如下: /* Name: 52coder Copyright: 52coder.net Author: 尘世中迷途小书童 Date: 15/05/15 11:49 Description: */ #include <stdio.h> void fun() { static int

【PHP内核学习】global关键字的解析过程分析

本文github地址: https://github.com/wusuopubupt/phpLib/blob/master/global%E5%85%B3%E9%94%AE%E5%AD%97%E7%9A%84%E8%A7%A3%E6%9E%90%E8%BF%87%E7%A8%8B%E5%88%86%E6%9E%90 |=-----------------------------------------------------------------------=| |=-------------

Java并发之synchronized关键字深度解析(二)

前言 本文继续[Java并发之synchronized关键字深度解析(一)]一文而来,着重介绍synchronized几种锁的特性. 一.对象头结构及锁状态标识 synchronized关键字是如何实现的给对象加锁?首先我们要了解一下java中对象的组成.java中的对象由3部分组成,第一部分是对象头,第二部分是实例数据,第三部分是对齐填充. 对齐填充:jvm规定对象的起始内存地址必须是8字节的整数倍,如果不够的话就用占位符来填充,此部分占位符就是对齐填充: 实例数据:实例数据是对象存储的真正有

java关键字static

static修饰符可以修饰变量.常量.方法和代码块,分别称为静态变量.静态常量.静态方法和静态代码块. 1.static变量 在java中,如果希望某个变量的值能被所有的对象共享,可以将该变量声明为静态变量(也叫类变量).静态变量在类装载时,只分配一块储存空间,所有此类的对象都可以操控此块储存空间,它为所有类实例提供共享的变量.当一个对象将该变量修改后,其他对象再使用该变量将会是改变后的数据.声明静态变量的语法如下: [权限控制符] static 成员变量类型 成员变量名: 访问静态变量的语法如

Java关键字--static

在Java中,将关键字static分为三部分进行讨论,分别为Java静态变量.Java静态方法.Java静态类 Java Static Variables Java instance variables are given separate memory for storage. If there is a need for a variable to be common to all the objects of a single java class, then the static mod

关键字static

这个简单的问题很少有人能回答完全.在C语言中,关键字static有三个明显的作用: 1)在函数体内,一个被声明为静态的变量在这一函数被调用过程中维持其值不变(该变量存放在静态变量区). 2) 在模块内(但在函数体外),一个被声明为静态的变量可以被模块内所用函数访问,但不能被模块外其它函数访问.它是一个本地的全局变量. 3) 在模块内,一个被声明为静态的函数只可被这一模块内的其它函数调用.那就是,这个函数被限制在声明它的模块的本地范围内使用. 大多数应试者能正确回答第一部分,一部分能正确回答第二部

java中的关键字static(静态变量)和final定义常量

1 package point; 2 3 class Point { 4 5 int x = 0; 6 int y = 0; 7 static int z = 100; // 定义静态变量z,类变量 8 static final double PI = 3.1415926;// final定义常量,它的值在运行时不能被改变 9 10 Point(int a, int b) { 11 //PI=3.1415926; 12 /* 13 * 当使用静态常量的时候,不能在构造函数中初始化, 因为静态时,

关键字static和const的作用

一.在C语言中,关键字static的作用: (1)设置变量的存储域,函数体内static变量的作用范围为该函数体,不同于auto变量,该变量的内存只被分配一次,因此其值在下次调用时仍维持上次的值: (2)限制变量的作用域,在模块内的static全局变量可以被模块内所用函数访问,但不能被模块外其它函数访问: (3)限制函数的作用域,在模块内的static函数只可被这一模块内的其它函数调用,这个函数的使用范围被限制在声明它的模块内:(4)在类中的static成员变量意味着它为该类的所有实例所共享,也

关键字static在c与c++中的区别。

关键字--static static主要有以下三个作用: 1. 局部静态变量(c/c++)2. 外部静态变量/函数(c/c++)3. 静态数据成员/成员函数 (c++特有)1.局部静态变量 静态局部变量,在函数里面定义的,就只能在这个函数里用了.由于被static修饰的变量只存在内存的静态区.所以即使函数运行结束,这个静态变量的值不会销毁,下次调用还能调用到这个值. 在C/C++中, 局部变量按照存储形式可分为三种auto, static, register. (1). 存储空间分配和生存周期不