java之static变量与全局、局部变量的区别

摘要:
静态变量与全局变量和局部变量之间的差异。全局变量的描述前面是静态的,以形成静态全局变量。静态函数具有不同于普通函数的作用域。外部全局变量、静态全局变量和静态局部变量的生存期都是“永久的”。唯一的区别是可见域。外部全局变量的可见区域是项目,静态全局变量的可视区域是文件,静态局部变量的可见区是块。

static变量与全局、局部变量的区别


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

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

 
  static函数与普通函数作用域不同。仅在本文件。只在当前源文件中使用的函数应该说明为内部函数(static),内部函数应该在当前源文件中说明和定义。对于可在当前源文件以外使用的函数,应该在一个头文件中说明,要使用这些函数的源文件要包含这个头文件

 
  static全局变量与普通的全局变量有什么区别:static全局变量只初使化一次,防止在其他文件单元中被引用;

  static局部变量和普通局部变量有什么区别:static局部变量只被初始化一次,下一次依据上一次结果值;

  static函数与普通函数有什么区别:static函数在内存中只有一份,普通函数在每个被调用中维持一份拷贝

 
  程序的局部变量存在于(堆栈)中,全局变量存在于(静态区 )中,动态申请数据存在于( 堆)中。

 
extern全局变量(用extern修饰的变量只是说明该变量在其他地方定义,所以在其他地方一定要用明确的定义如int a,并且不能用static修饰)、static全局变量和static局部变量的生存期都是“永久”,区别只是可见域不同。extern全局变量可见区域是工程,static全局变量可见区域是文件,而static局部变量的可见区域是块。

从代码维护角度来看,对extern变量的修改可能会影响所有代码,对static全局变量的修改可能影响一个文件中的代码,而对static变量的修改可能影响一个块的代码;因此在选择变量类型时,优先级是static局部>static全局>extern全局。但它们有着共同的缺点:使用了这些类型变量的函数将是不可重入的,不是线程安全的。在C/C++标准库中有很多函数都使用了static局部变量,目前的实现中都为它们提供了两套代码,单线程版本使用static变量而多线程版本使用“线程全局变量”,比如rand,strtok等。

 
一个进程可用内存空间为4G,可分在存放静态数据,代码,系统内存,堆,栈等。.活动记录一般存放调用参数、返回地址等内容。堆和栈最大的区别在于堆是由低地址向高地址分配内存,而栈是由高向低。全局和静态数据存放在全局数据区,其余的在栈中,用malloc 或 new 分配的内存位于堆中。一般来说栈在低地址,堆位于高地址。

static的全部用法

        要理解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是按照程序的生命周期来分配释放变量的,而不是变量自己的生命周期;所以,像这样的例子:

       void func()

       {

              int a;

              static int b;

}

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

static的全部用法:

1.类的静态成员:

class A

{

private:

       static int s_value;

};

在cpp中必须对它进行初始化:

int A::s_value = 0;          // 注意,这里没有static的修饰!

类的静态成员是该类所有实例的共用成员,也就是在该类的范畴内是个全局变量,也可以理解为是一个名为A::s_value的全局变量,只不过它是带有类安全属性的;道理很简单,因为它是在程序初始化的时候分配的,所以只分配一次,所以就是共用的;

类的静态成员必须初始化,道理也是一样的,因为它是在程序初始化的时候分配的,所以必须有初始化,类中只是声明,在cpp中才是初始化,你可以在初始化的代码上放个断点,在程序执行main的第一条语句之前就会先走到那;如果你的静态成员是个类,那么就会调用到它的构造函数;

2.类的静态函数:

class A

{

private:

       static void func(int value);

};

实现的时候也不需要static的修饰,因为static是声明性关键字;

类的静态函数是在该类的范畴内的全局函数,不能访问类的私有成员,只能访问类的静态成员,不需要类的实例即可调用;实际上,它就是增加了类的访问权限的全局函数:void A::func(int);

类的静态成员函数可以继承和覆盖,但无法是需函数;

3.只在cpp内有效的全局变量:

在cpp文件的全局范围内声明:

static int g_value = 0;

        这个变量的含义是在该cpp内有效,但是其他的cpp文件不能访问这个变量;如果有两个cpp文件声明了同名的全局静态变量,那么他们实际上是独立的两个变量;

        如果不使用static声明全局变量:

       int g_value = 0;

        那么将无法保证这个变量不被别的cpp共享,也无法保证一定能被别的cpp共享,因为要让多个cpp共享一个全局变量,应将它声明为extern(外部)的;也有可能编译会报告变量被重复定义;总之不建议这样的写法,不明确这个全局变量的用法;

        如果在一个头文件中声明:

       static int g_vaule = 0;

        那么会为每个包含该头文件的cpp都创建一个全局变量,但他们都是独立的;所以也不建议这样的写法,一样不明确需要怎样使用这个变量,因为只是创建了一组同名而不同作用域的变量;

        这里顺便说一下如何声明所有cpp可共享的全局变量,在头文件里声明为extern的:

       extern int g_value;        // 注意,不要初始化值!

        然后在其中任何一个包含该头文件的cpp中初始化(一次)就好:

       int g_value = 0;     // 初始化一样不要extern修饰,因为extern也是声明性关键字;

        然后所有包含该头文件的cpp文件都可以用g_value这个名字访问相同的一个变量;

      

       4.只在cpp内有效的全局函数:

        在cpp内声明:

       static void func();

        函数的实现不需要static修饰,那么这个函数只可在本cpp内使用,不会同其他cpp中的同名函数引起冲突;道理和如果不使用static会引起的问题和第3点一样;不要在头文件中声明static的全局函数,不要在cpp内声明非static的全局函数,如果你要在多个cpp中复用该函数,就把它的声明提到头文件里去,否则在cpp内部声明需要加上static修饰;在C语言中这点由为重要!

       5.局部静态变量:

       void func()

       {

              static int s_value = 0;

}

在其作用域内的静态变量,多线程情况下将访问同一个变量,由于函数调用结束仍保持静态变量的内存,因而可以作为一些动态变量返回值的存放位置:

const char * func()

{

       string s = “ABCD”;

       static char msg[256];

       strcpy(msg, s);

       return msg;

}

不能直接返回s,因为它是动态的;否则将导致一个很经典的内存问题(不用我细说了吧),通常情况下我们返回一个常量字符串:return “ABCD”;其实是因为”ABCD”就是static char[]类型的(所以才能没有问题);

 

免责声明:文章转载自《java之static变量与全局、局部变量的区别》仅用于学习参考。如对内容有疑问,请及时联系本站处理。

上篇Unity打开VisualStudio一直显示hold on的解决办法(自醒)WPF-带有GridView的ListView样式下篇

宿迁高防,2C2G15M,22元/月;香港BGP,2C5G5M,25元/月 雨云优惠码:MjYwNzM=

相关文章

ios 字典没有值的问题

我遇到这样一个问题:初始化一个字典,初始化的时候,给三个key-value;但是,打印字典的时候,只有第一个元素有值,后两个没有值。  NSDictionary *dict = [NSDictionary dictionaryWithObjectsAndKeys:_messageString,@"message",_photoImage,@"photo",...

cke点击时初始化编辑器后光标恢复的方法

业务场景 1、使用了CKEDITOR编辑器 2、文本是使用contenteditable="true"的div容器 3、点击文本时才初始化编辑器 4、问题:编辑器初始化后光标会重置到开始处,如何将光标重置到点击处 解决方案 1、在点击文本的时候,在点击文本的时候,获取range信息和 endContainer与endOffset; try {...

多线程下C#如何保证线程安全?

  多线程编程相对于单线程会出现一个特有的问题,就是线程安全的问题。所谓的线程安全,就是如果你的代码所在的进程中有多个线程在同时运行,而这些线程可能会同时运行这段代码。如果每次运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的。 线程安全问题都是由全局变量及静态变量引起的。   为了保证多线程情况下,访问静态变量的安全,可以用锁机制来...

Debug 和 Release 编译方式的本质区别

一、Debug 和 Release 编译方式的本质区别       Debug 通常称为调试版本,它包含调试信息,并且不作任何优化,便于程序员调试程序。Release 称为发布版本,它往往是进行了各种优化,使得程序在代码大小和运行速度上都是最优的,以便用户很好地使用。       Debug 和 Release 的真正秘密,在于一组编译选项。下面列出了分别...

golang反射

要点 1.变量 2.反射 3.结构体反射 4.反射总结以及应用场景 一、变量介绍 1.变量的内在机制 A、类型信息,这部分是元信息,是预定义好的 B、值类型,这部分是程序运行过程中,动态改变的 var arr [10]int arr[0] = 10 arr[1] = 20 arr[2] = 30 arr[3] = 40 type Animal s...

IVR交互式语音应答系统

ZingServ平台IVR子系统介绍 IVR系统的中文译名为"交互式语音应答系统",然而,真正的IVR系统早已超出这种中文译名所描述的功能范围--IVR是一种功能强大的电话自动服务系统。对用户来说,只有理解了这一点,才能真正了解IVR系统的价值;而对产品提供商来说,也只有理解了这一点,才能真正明确产品的应用范围和设计目标。  在ZingServ一体化呼叫中...