google了近三页的关于C语言中static的内容,发现可用的信息很少,要么长篇大论不知所云要么在关键之处几个字略过,对于想挖掘底层原理的初学者来说参考性不是很大。所以,我这篇博文博采众家之长,把互联网上的资料整合归类,并亲手编写程序验证之。

C语言代码是以文件为单位来组织的,在一个源程序的所有源文件中,一个外部变量(注意不是局部变量)或者函数只能在一个源程序中定义一次,如果有重复定义的话编译器就会报错。伴随着不同源文件变量和函数之间的相互引用以及相互独立的关系,产生了extern和static关键字。

下面,详细分析一下static关键字在编写程序时有的三大类用法:

一,static全局变量

我们知道,一个进程在内存中的布局如图1所示:

其中.text段保存进程所执行的程序二进制文件,.data段保存进程所有的已初始化的全局变量,.bss段保存进程未初始化的全局变量(其他段中还有很多乱七八糟的段,暂且不表)。在进程的整个生命周期中,.data段和.bss段内的数据时跟整个进程同生共死的,也就是在进程结束之后这些数据才会寿终就寝。

当一个进程的全局变量被声明为static之后,它的中文名叫静态全局变量。静态全局变量和其他的全局变量的存储地点并没有区别,都是在.data段(已初始化)或者.bss段(未初始化)内,但是它只在定义它的源文件内有效,其他源文件无法访问它。所以,普通全局变量穿上static外衣后,它就变成了新娘,已心有所属,只能被定义它的源文件(新郎)中的变量或函数访问。

以下是一些示例程序

file1.h如下:

[cpp] view plaincopy print?
  1. #include <stdio.h>
  2. void printStr();

#include <stdio.h>void printStr();

我们在file1.c中定义一个静态全局变量hello, 供file1.c中的函数printStr访问.

[cpp] view plaincopy print?
  1. #include "file1.h"
  2. static char* hello = "hello cobing!";
  3. void printStr()
  4. {
  5. printf("%s\n", hello);
  6. }

#include "file1.h"static char* hello = "hello cobing!";void printStr()
{printf("%s\n", hello);
}

file2.c是我们的主程序所在文件,file2.c中如果引用hello会编译出错

[cpp] view plaincopy print?
  1. #include "file1.h"
  2. int main()
  3. {
  4. printStr();
  5. printf("%s\n", hello);
  6. return 0;
  7. }

#include "file1.h"int main()
{printStr();printf("%s\n", hello);return 0;
}

报错如下:

[liujx@server235 static]$ gcc -Wall file2.c file1.c -o file2
file2.c: In function ‘main’:
file2.c:6: 错误:‘hello’ 未声明 (在此函数内第一次使用)
file2.c:6: 错误:(即使在一个函数内多次出现,每个未声明的标识符在其
file2.c:6: 错误:所在的函数内只报告一次。)

如果我们将file2.c改为下面的形式:

[cpp] view plaincopy print?
  1. #include "file1.h"
  2. int main()
  3. {
  4. printStr();
  5. return 0;
  6. }

#include "file1.h"int main()
{printStr();return 0;
}

则会顺利编译连接。

运行程序后的结果如下:
[liujx@server235 static]$ gcc -Wall file2.c file1.c -o file2
[liujx@server235 static]$ ./file2
hello cobing!

上面的例子中,file1.c中的hello就是一个静态全局变量,它可以被同一文件中的printStr调用,但是不能被不同源文件中的file2.c调用。

二,static局部变量

普通的局部变量在栈空间上分配,这个局部变量所在的函数被多次调用时,每次调用这个局部变量在栈上的位置都不一定相同。局部变量也可以在堆上动态分配,但是记得使用完这个堆空间后要释放之。

static局部变量中文名叫静态局部变量。它与普通的局部变量比起来有如下几个区别:

1)位置:静态局部变量被编译器放在全局存储区.data(注意:不在.bss段内,原因见3)),所以它虽然是局部的,但是在程序的整个生命周期中存在。

2)访问权限:静态局部变量只能被其作用域内的变量或函数访问。也就是说虽然它会在程序的整个生命周期中存在,由于它是static的,它不能被其他的函数和源文件访问。

3)值:静态局部变量如果没有被用户初始化,则会被编译器自动赋值为0,以后每次调用静态局部变量的时候都用上次调用后的值。这个比较好理解,每次函数调用静态局部变量的时候都修改它然后离开,下次读的时候从全局存储区读出的静态局部变量就是上次修改后的值。
以下是一些示例程序:

file1.h的内容和上例中的相同,file1.c的内容如下:

[cpp] view plaincopy print?
  1. #include "file1.h"
  2. void printStr()
  3. {
  4. int normal = 0;
  5. static int stat = 0;    //this is a static local var
  6. printf("normal = %d ---- stat = %d\n",normal, stat);
  7. normal++;
  8. stat++;
  9. }

#include "file1.h"void printStr()
{int normal = 0;static int stat = 0;  //this is a static local varprintf("normal = %d ---- stat = %d\n",normal, stat);normal++;stat++;
}

为了便于比较,我定义了两个变量:普通局部变量normal和静态局部变量stat,它们都被赋予初值0;

file2.c中调用file1.h:

[cpp] view plaincopy print?
  1. #include "file1.h"
  2. int main()
  3. {
  4. printStr();
  5. printStr();
  6. printStr();
  7. printStr();
  8. printf("call stat in main: %d\n",stat);
  9. return 0;
  10. }

#include "file1.h"int main()
{printStr();printStr();printStr();printStr();printf("call stat in main: %d\n",stat);return 0;
}

这个调用会报错,因为file2.c中引用了file1.c中的静态局部变量stat,如下:

[liujx@server235 static]$ gcc -Wall file2.c file1.c -o file2
file2.c: In function ‘main’:
file2.c:9: 错误:‘stat’ 未声明 (在此函数内第一次使用)
file2.c:9: 错误:(即使在一个函数内多次出现,每个未声明的标识符在其
file2.c:9: 错误:所在的函数内只报告一次。)

编译器说stat未声明,这是因为它看不到file1.c中的stat,下面注掉这一行:

[cpp] view plaincopy print?
  1. #include "file1.h"
  2. int main()
  3. {
  4. printStr();
  5. printStr();
  6. printStr();
  7. printStr();
  8. //  printf("call stat in main: %d\n",stat);
  9. return 0;
  10. }

#include "file1.h"int main()
{printStr();printStr();printStr();printStr();
//  printf("call stat in main: %d\n",stat);return 0;
}

[liujx@server235 static]$ gcc -Wall file2.c file1.c -o file2
[liujx@server235 static]$ ./file2
normal = 0 ---- stat = 0
normal = 0 ---- stat = 1
normal = 0 ---- stat = 2
normal = 0 ---- stat = 3
运行如上所示。可以看出,函数每次被调用,普通局部变量都是重新分配,而静态局部变量保持上次调用的值不变。

需要注意的是由于static局部变量的这种特性,使得含静态局部变量的函数变得不可重入,即每次调用可能会产生不同的结果。这在多线程编程时可能会成为一种隐患。需要多加注意。

三,static函数
              相信大家还记得C++面向对象编程中的private函数,私有函数只有该类的成员变量或成员函数可以访问。在C语言中,也有“private函数”,它就是接下来要说的static函数,完成面向对象编程中private函数的功能。

当你的程序中有很多个源文件的时候,你肯定会让某个源文件只提供一些外界需要的接口,其他的函数可能是为了实现这些接口而编写,这些其他的函数你可能并不希望被外界(非本源文件)所看到,这时候就可以用static修饰这些“其他的函数”。

所以static函数的作用域是本源文件,把它想象为面向对象中的private函数就可以了。

下面是一些示例:

file1.h如下:

[cpp] view plaincopy print?
  1. #include <stdio.h>
  2. static int called();
  3. void printStr();

#include <stdio.h>static int called();
void printStr();

file1.c如下:

[cpp] view plaincopy print?
  1. #include "file1.h"
  2. static int called()
  3. {
  4. return 6;
  5. }
  6. void printStr()
  7. {
  8. int returnVal;
  9. returnVal = called();
  10. printf("returnVal=%d\n",returnVal);
  11. }

#include "file1.h"static int called()
{return 6;
}
void printStr()
{int returnVal;returnVal = called();printf("returnVal=%d\n",returnVal);
}

file2.c中调用file1.h中声明的两个函数,此处我们故意调用called():

[cpp] view plaincopy print?
  1. #include "file1.h"
  2. int main()
  3. {
  4. int val;
  5. val = called();
  6. printStr();
  7. return 0;
  8. }

#include "file1.h"int main()
{int val;val = called();printStr();return 0;
}

编译时会报错:

[liujx@server235 static]$ gcc -Wall file2.c file1.c -o file2
file1.h:3: 警告:‘called’ 使用过但从未定义
/tmp/ccyLuBZU.o: In function `main':
file2.c:(.text+0x12): undefined reference to `called'
collect2: ld 返回 1
因为引用了file1.h中的static函数,所以file2.c中提示找不到这个函数:undefined reference to 'called'

下面修改file2.c:

[cpp] view plaincopy print?
  1. #include "file1.h"
  2. int main()
  3. {
  4. printStr();
  5. return 0;
  6. }

#include "file1.h"int main()
{printStr();return 0;
}

编译运行:

[liujx@server235 static]$ gcc -Wall file2.c file1.c -o file2
[liujx@server235 static]$ ./file2
returnVal=6

static函数可以很好地解决不同原文件中函数同名的问题,因为一个源文件对于其他源文件中的static函数是不可见的。

有疏漏的地方望各位多多指教~~

C语言中static详细分析相关推荐

  1. C语言中的static 详细分析

    google了近三页的关于C语言中static的内容,发现可用的信息很少,要么长篇大论不知所云要么在关键之处几个字略过,对于想挖掘底层原理的初学者来说参考性不是很大.所以,我这篇博文博采众家之长,把互 ...

  2. C语言中的static 详细分析 2014-10-11 15:15 143人阅读 评论(0) 收藏...

    转自:http://blog.csdn.net/keyeagle/article/details/6708077 看到这篇文章针对C语言的static关键字解释的比较好,这里转载过来.谢谢原作者分享. ...

  3. C语言中static变量详解

    google了近三页的关于C语言中static的内容,发现可用的信息很少,要么长篇大论不知所云要么在关键之处几个字略过,对于想挖掘底层原理的初学者来说参考性不是很大.所以,我这篇博文博采众家之长,把互 ...

  4. c语言中static变量

    c语言中static变量 2009-06-30 20:33:24|  分类:默认分类 |  标签:|字号大中小 订阅 static静态变量声明符. 在声明它的程序块,子程序块或函数内部有效,值保持,在 ...

  5. C语言中static的作用及C语言中使用静态函数有何好处

    转自:http://www.jb51.net/article/74830.htm 在C语言中,static的作用有三条:一是隐藏功能,二是保持持久性功能,三是默认初始化为0. 在C语言中,static ...

  6. C语言中static关键字的作用

    在C语言中static的作用如下 第一.在修饰变量的时候,static修饰的静态局部变量只执行一次,而且延长了局部变量的生命周期,直到程序运行结束以后才释放.  第二.static修饰全局变量的时候, ...

  7. 【✊基础不牢,地动山摇のC语言中static关键字✊】

    C语言中static关键字 用static声明限定外部变量与函数,可以将其后声明的对象的作用域限定为被编译源文件的剩余部分.要降对象指定为静态存储,可以在正常的声明之前加上关键字static作为前缀. ...

  8. 以下关于c语言中static和const,c语言中static const作用

    c语言中static const作用 (2012-06-21 07:51:08) 标签: it 关键字static: 1. 在函数体内,一个被声明为静态的变量在这一个函数被调用的过程中维持其值不变. ...

  9. c语言中{的作用,C语言中Static和Const关键字的作用

    C语言中Static和Const关键字的作用 程序的局部变量存在于(堆栈)中,全局变量存在于(静态区 )中,动态申请数据存在于(堆)中.那么关于C语言中Static和Const关键字的作用,你了解多少 ...

最新文章

  1. 3亿人养老靠机器人?这家公司要在2030年实现,有谱
  2. Greenplum——升级的分布式PostgresSQL
  3. python与人工智能编程-Python是人工智能和机器学习的最佳编程语言,证据在此!...
  4. python一次性读取整个文件-python-文件中的行是否读取整个文件
  5. MFC 加载并显示图片的四种方法
  6. 获取qt保存对话框中输入文本_PyQt5 输入对话框QInputDialog
  7. linux 支持的字体命令,Linux设置显示中文和字体
  8. android如何导入活动,关于android:如何将活动值传递给另一个活动(Kotlin)
  9. Leetcode 9. 回文数(Palindrome Number)
  10. ConvMixer:7行PyTorch代码实现的网络,就能在ImageNet上达到80%+的精度!
  11. mysql 查询包含1或者2_Mysql:同一个表上有2个不同的查询,包含count和group by
  12. C# WinForm开发系列 - 开篇
  13. TCP/IP:SCTP报文格式
  14. Coinbase在上市前选择在Reddit线上路演
  15. word排版案例报告_原来按下这个键,10秒EXCEL与Word格式就能互相转换,涨知识了...
  16. 《在近端对回传音频的检测和抑制》笔记
  17. java版mc植物生长条件_植物生长三大必要条件
  18. c语言c11标准 下载,【整理】C语言的各种版本:C89,AMD1,C99,C11
  19. Android 鼠标键值列表,安卓键盘键值对照表
  20. 深入分析AIL语言及init.rc文件

热门文章

  1. 秒杀多线程第十篇 生产者消费者问题 (续)
  2. Python 学习散记
  3. HDOJ 1175 连连看 DFS
  4. 实现Windows Phone、Android和iOS平台的统一硬件访问
  5. PXE网络装机之centos7(批量自动装机)
  6. CIA困局:天下再无007,AI识别下无处遁行的“特工”们
  7. 编译Android 4.0 ICS注意事项
  8. 恢复删除的Linux文件
  9. dedecms二级菜单中判断子菜单标签的方法
  10. BGP信息类型和分组公共首部