C语言中的stati c详细分析 标签语言cfi leserver编译器reference
2011-08-2212:0432786人阅读评论(34)收藏举报分类
C 4
版权声明本文为博主原创文章未经博主允许不得转载。g oog le了近三页的关于C语言中s tati c的内容发现可用的信息很少要么长篇大论不知所云要么在关键之处几个字略过对于想挖掘底层原理的初学者来说参考性不是很大。所以我这篇博文博采众家之长把互联网上的资料整合归类并亲手编写程序验证之。
C语言代码是以文件为单位来组织的在一个源程序的所有源文件中一个外部变量注意不是局部变量或者函数只能在一个源程序中定义一次如果有重复定义的话编译器就会报错。伴随着不同源文件变量和函数之间的相互引用以及相互独立的关系产生了exte rn和s tatic关键字。
下面详细分析一下s tati c关键字在编写程序时有的三大类用法
一 s tatic全局变量
我们知道一个进程在内存中的布局如图1所示
其中.text段保存进程所执行的程序二进制文件 .data段保存进程所有的已初始化的全局变量 .bss段保存进程未初始化的全局变量其他段中还有很多乱七八糟的段暂且不表 。在进程的整个生命周期中 .data段和.bss段内的数据时跟整个进程同生共死的也就是在进程结束之后这些数据才会寿终就寝。
当一个进程的全局变量被声明为static之后它的中文名叫静态全局变量。静态全局变量和其他的全局变量的存储地点并没有区别都是在.data段已初始化或者.bss段未初始化 内但是它只在定义它的源文件内有效其他源文件无法访问它。所以普通全局变量穿上static外衣后它就变成了新娘 已心有所属只能被定义它的源文件新郎中的变量或函数访问。
以下是一些示例程序fi le1 h如下
[cpp]view plaincopy
我们在fi le1 c中定义一个静态全局变量hello,供fi le1 c中的函数printStr访问
[cpp]view plaincopy
! ";
;fi le2c是我们的主程序所在文件 fi le2c中如果引用hel lo会编译出错
[cpp]view plaincopy
报错如下
[l iujx@server235static]$gcc-Wal l fi le2 cfi le1 c-o fi le2fi le2c: Infunction„main‟ :fi le2c:6:错误 „hel lo‟未声明(在此函数内第一次使用)fi le2c:6:错误 (即使在一个函数内多次出现每个未声明的标识符在其fi le2c:6:错误所在的函数内只报告一次。 )
如果我们将fi le2 c改为下面的形式
[cpp]view plaincopy
则会顺利编译连接。
运行程序后的结果如下
[l iujx@server235static]$gcc-Wal l fi le2 cfi le1 c-o fi le2
[l iujx@server235 static]$ /fi le2hel lo cobing!
上面的例子中 fi le1 c中的hel lo就是一个静态全局变量它可以被同一文件中的printStr调用但是不能被不同源文件中的fi le2 c调用。
二 s tatic局部变量
普通的局部变量在栈空间上分配这个局部变量所在的函数被多次调用时每次调用这个局部变量在栈上的位置都不一定相同。局部变量也可以在堆上动态分配但是记得使用完这个堆空间后要释放之。static局部变量中文名叫静态局部变量。它与普通的局部变量比起来有如下几个区别
1 位置静态局部变量被编译器放在全局存储区.data 注意不在.bss段内原因见3 所以它虽然是局部的但是在程序的整个生命周期中存在。
2访问权限静态局部变量只能被其作用域内的变量或函数访问。也就是说虽然它会在程序的整个生命周期中存在由于它是static的它不能被其他的函数和源文件访问。
3值静态局部变量如果没有被用户初始化则会被编译器自动赋值为0 以后每次调用静态局部变量的时候都用上次调用后的值。这个比较好理解每次函数调用静态局部变量的时候都修改它然后离开下次读的时候从全局存储区读出的静态局部变量就是上次修改后的值。
以下是一些示例程序fi le1 h的内容和上例中的相同 fi le1c的内容如下
[cpp]view plaincopy
为了便于比较我定义了两个变量普通局部变量norma l和静态局部变量stat它们都被赋予初值0fi le2c中调用fi le1 h
[cpp]view plaincopy
这个调用会报错 因为fi le2 c中引用了fi le1 c中的静态局部变量stat如下
[l iujx@server235static]$gcc-Wal l fi le2 cfi le1 c-o fi le2fi le2c: Infunction„main‟ :fi le2 c:9:错误 „stat‟未声明(在此函数内第一次使用)fi le2c:9:错误 (即使在一个函数内多次出现每个未声明的标识符在其fi le2c:9:错误所在的函数内只报告一次。 )
编译器说stat未声明这是因为它看不到fi le1 c中的stat下面注掉这一行
[cpp]view plaincopy
[l iujx@server235static]$gcc-Wal l fi le2 cfi le1 c-o fi le2
[l iujx@server235 static]$ /fi le2norm al=0----stat=0norm al=0----stat=1norm al=0----stat=2norm al=0----stat=3
运行如上所示。可以看出 函数每次被调用普通局部变量都是重新分配而静态局部变量保持上次调用的值不变。
需要注意的是由于static局部变量的这种特性使得含静态局部变量的函数变得不可重入 即每次调用可能会产生不同的结果。这在多线程编程时可能会成为一种隐患。需要多加注意。
三 s tatic函数
相信大家还记得C++面向对象编程中的private函数私有函数只有该类的成员变量或成员函数可以访问。在C语言中也有“private函数”它就是接下来要说的static函数完成面向对象编程中private函数的功能。
当你的程序中有很多个源文件的时候你肯定会让某个源文件只提供一些外界需要的接口其他的函数可能是为了实现这些接口而编写这些其他的函数你可能并不希望被外界非本源文件所看到这时候就可以用s tati c修饰这些“其他的函数”。
所以static函数的作用域是本源文件把它想象为面向对象中的private函数就可以了。
下面是一些示例fi le1 h如下
[cpp]view plaincopy
fi le1c如下
[cpp]view plaincopy
fi le2c中调用fi le1 h中声明的两个函数此处我们故意调用cal led():
[cpp]view plaincopy
编译时会报错
[l iujx@server235static]$gcc-Wal l fi le2 cfi le1 c-o fi le2fi le1 h:3:警告 „cal led‟使用过但从未定义
/tmp/ccyLuBZU o: In function`main' :fi le2 c:( text+0x12):undefined reference to`cal led'co l l ect2: l d返回1
因为引用了fi le1 h中的static函数所以fi le2 c中提示找不到这个函数:undefined reference to'cal led'
下面修改fi le2 c:
[cpp]view plaincopy
编译运行
[l iujx@server235static]$gcc-Wal l fi le2 cfi le1 c-o fi le2
[l iujx@server235 static]$ /fi le2returnVal=6s tatic函数可以很好地解决不同原文件中函数同名的问题 因为一个源文件对于其他源文件中的s tati c函数是不可见的。有疏漏的地方望各位多多指教~~
美得云成立于2021年,是一家云产品管理服务商(cloud)专业提供云计算服务、DDOS防护、网络安全服务、国内海外数据中心托管租用等业务、20000+用户的选择,43800+小时稳定运行香港特价将军澳CTG+CN2云服务器、采用高端CPU 优质CN2路线 SDD硬盘。香港CTG+CN22核2G3M20G数据盘25元点击购买香港CTG+CN22核2G5M30G数据盘39元点击购买香港CTG+CN...
onevps最新消息,为了更好服务中国区用户:1、网站支付方式新增了支付宝,即将增加微信;原信用卡、PayPal方式不变;(2)可以切换简体中文版网站,在网站顶部右上角找到那个米字旗,下拉可以换中国简体版本。VPS可选机房有:中国(香港)、新加坡、日本(东京)、美国(纽约、洛杉矶)、英国(伦敦)、荷兰(阿姆斯特丹)、瑞士(苏黎世)、德国(法兰克福)、澳大利亚(悉尼)。不管你的客户在亚太区域、美洲区...
官方网站:点击访问青云互联官网优惠码:五折优惠码:5LHbEhaS (一次性五折,可月付、季付、半年付、年付)活动方案:的套餐分为大带宽限流和小带宽不限流两种套餐,全部为KVM虚拟架构,而且配置都可以弹性设置1、洛杉矶cera机房三网回程cn2gia 洛杉矶cera机房  ...