关于局部变量、全局变量与静态全局变量和静态局部变量的比较与分析和拓展:
首先要明白的是局部变量与其它三个有本质区别。
局部变量,定义在函数内,作用域为定义位置,到所在大括号结束。 每次执行到定义语句的时候,系统为该变量分配内存,当作用域结束后,内存被释放。
而另外三个,都是在系统运行时就分配内存,在运行过程中都不会释放,直到程序结束。
也就是说,局部变量和其它三个,最明显的区别:
1 生命周期不同
2 分配时间不同
3 分配位置不同。
而另外三个,在这几项上是相同的,之间的区别在于作用域。
全局变量,作用域为整个项目,不管在哪个文件中,只要声明后都可以使用,当然,其他不包含全局变量定义的源文件需要用extern 关键字再次声明这个全局变量。
静态全局变量也具有全局作用域,他与全局变量的区别在于如果程序包含多个文件的话,他作用于定义它的文件里,不能作用到其他文件里,即被static关键字修饰过的变量具有文件作用域。这样即使两个不同的源文件都定义了相同的静态全局变量,他们也是不同的变量。
而静态局部变量,作用域与普通局部变量一样,都是定义位置到所在大括号结束。静态局部变量具有局部作用域,它只被初始化一次,自从第一次被初始化直到程序运行结束都一直存在,他和全局变量的区别在于全局变量对所有的函数都是可见的,而静态局部变量只对定义自己的函数体始终可见。
需要注意的是:
1>全局变量,静态局部变量,静态全局变量都在静态存储区分配空间,而局部变量在栈里分配空间。
2>全局变量本身就是静态存储方式, 静态全局变量当然也是静态存储方式。这两者在存储方式上并无不同。这两者的区别虽在于非静态全局变量的作用域是整个源程序,当一个源程序由多个源文件组成时,非静态的全局变量在各个源文件中都是有效的。而静态全局变量则限制了其作用域,即只在定义该变量的源文件内有效,在同一源程序的其它源文件中不能使用它。由于静态全局变量的作用域局限于一个源文件内,只能为该源文件内的函数公用,因此可以避免在其它源文件中引起错误。
注释:static修饰局部变量说明
1)静态局部变量属于静态存储类别,在静态存储区分配存储单元。 在程序整个运行期间都不释放。自动变量(即动态局部变量)属于动态存储类别,占动态存储区空间,函数调用之后即 释放。
2)对静态局部变量,是在编译时候赋初值的,即只付出至一次,在程序运行时它已由初值。以 后每次调用的时候不再重新赋初值而只是保留上次函数调用结束时的值。而对自动变量赋初值,不是在编译的时候进行的,而是在函数调用时进行。每调用一次函数,重新给一个初值。 相当于执行一次赋值语句。静态变量会被放在程序的静态数据存储区里,这样可以在下一次调用的时候还可以保持原来的赋值。这一点是他与堆栈变量和堆变量的区别
3)如在定义局部变量时不赋初值的话,则编译器对静态变量默认赋值为0(整型)或者空字符(字符型) ,对动态变量只分配内存空间,并不会赋默认值,所以里面的值是随机的。
4)静态局部变量在函数调用结束之后仍然是存在的,但是作用域不变,其他函数依然不能引用。
关于static功能的拓展:
1.static与 extern对函数的作用:
函数的本质是全局的,写函数就是用来调用的,但是有些时候不想让其他文件访问指定函数。
根据函数能否被其他文件调用,将函数分为了内部函数和外部函数。
内部函数:只能被本文件中其他函数所调用,成为内部函数。在定义函数的时候使用static修饰就可以了。
外部函数:能被其他文件访问,定义的时候在最左边加关键字extern显示表示为外部函数,不加则表示隐含的extern。
2.函数中必须要使用static变量的情况:当某函数的返回值为指针类型时,则必须是static的局部变量的地址作为返回值,若为auto类型(参考//注释:),则返回为其指针。局部变量在函数调用后就销毁了,虽然地址还在,但是那个地址存的数据是无法保证安全性的,因此如果想指向被调函数中的参数,最好将其定义为静态的~
//注释:
auto在函数中的局部变量,如果不专门声明为static存储类别,都是动态分配存储空间的,数据存储在动态存储区中。函数中的形参和函数中的变量都是这种类型的。在调用该函数的时候系统会给分配内存空间,函数调用完毕之后,自动释放这些存储空间。因此这类局部变量称为自动变量。
函数中,定义auto int a = 10,跟int a = 10等价Register
关键字regiter请求编译器尽可能的将变量存在CPU的寄存器中。有以下几点注意的地方。
register变量必须是能被CPU寄存器所接受的类型,这通常意味着register变量必须是一个单个的值,并且其长度应小于或等于整型的长度。但是,有些机器的寄存器也能存放浮点数。
register变量可能不存放在内存中,所以不能用取址符运算符“& ”。 只有局部变量和形参可以作为register变量,全局变量不行。 静态变量不能定义为register。