关于C语言中定义static变量的解释

       在C语言中static类型的变量的意思其实和它的本意差不多,表示“静态”或者“全局”的意思,用来修饰变量和函数。经static修饰过后的变量或者函数的作用域或者存储域会发生变化,而由static修饰的变量在初始值方面也会表现出static关键字的优势。

C语言中程序的内存分布:

    1)正文段 —— CPU执行的机器指令部分;一个程序只有一个副本;只读,防止程序由于意外事故而修改自身指令;

    2)初始化数据段(数据段)—— 在程序中所有赋了初值的全局变量,存放在这里。

    3)非初始化数据段(bss段)—— 在程序中没有初始化的全局变量;内核将此段初始化为0。

    4)栈 —— 增长方向:自顶向下增长;自动变量以及每次函数调用时所需要保存的信息(返回地址;环境信息)。

 5)堆 —— 动态存储区。是向高地址扩展的数据类型,是自下向上的扩展方式。

(此图来源于网络)

static修饰的全局变量

  在全局变量之前加上关键字static,全局变量就被定义成为一个全局静态变量。

      1)内存中的位置:静态存储区(静态存储区在整个程序运行期间都存在)

      2)初始化:未经初始化的全局静态变量会被程序自动初始化为0(自动对象的值是任意的,除非他被显示初始化)

      3)作用域:全局静态变量在声明他的文件之外是不可见的。准确地讲从定义之处开始到文件结尾。

       定义全局静态变量的好处:

      <1>不会被其他文件所访问,修改

      <2>其他文件中可以使用相同名字的变量,不会发生冲突。

  注:当static用来修饰全局变量的时候,它就改变了全局变量的作用域(在声明他的文件之外是不可见的),但是没有改变它的存放位置,还是在静态存储区中。

static修饰的局部变量

       在局部变量之前加上关键字static,局部变量就被定义成为一个局部静态变量。

      1)内存中的位置:静态存储区

      2)初始化:未经初始化的局部静态变量会被程序自动初始化为0(自动对象的值是任意的,除非他被显示初始化)

      3)作用域:作用域仍为局部作用域,当定义它的函数或者语句块结束的时候,作用域随之结束。

  注:当static用来修饰局部变量的时候,它就改变了局部变量的存储位置,从原来的栈中存放改为静态存储区。但是局部静态变量在离开作用域之后,并没有被销毁,而是仍然驻留在内存当中,直到程序结束,只不过我们不能再对他进行访问。

Static修饰的函数

     在函数的返回类型前加上关键字static,函数就被定义成为静态函数。函数的定义和声明默认情况下是extern的,但静态函数只是在声明他的文件中可见,不能被其他文件所用。

定义静态函数的好处:
<1> 其他文件中可以定义相同名字的函数,不会发生冲突 

<2> 静态函数不能被其他文件所用。


       存储说明符auto,register,extern,static,对应两种存储期:自动存储期和静态存储期。auto和register对应自动存储期。具有自动存储期的变量在进入声明该变量的程序块时被建立,它在该程序块活动时存在,退出该程序块时撤销。
  关键字extern和static用来说明具有静态存储期的变量和函数。用static声明的局部变量具有静态存储持续期(static storage duration),或静态范围(static extent)。虽然他的值在函数调用之间保持有效,但是其名字的可视性仍限制在其局部域内。静态局部对象在程序执行到该对象的声明处时被首次初始化。

总结

(1)第一个作用:隐藏。

(2)static的第二个作用是保持变量内容的持久。存储在静态数据区的变量会在程序刚开始运行时就完成初始化,也是唯一的一次初始化。共有两种变量存储在静态存储区:全局变量和static变量,只不过和全局变量比起来,static可以控制变量的可见范围,说到底static还是用来隐藏的。

(3)static的第三个作用是默认初始化为0。其实全局变量也具备这一属性,因为全局变量也存储在静态数据区。在静态数据区,内存中所有的字节默认值都是0x00,某些时候这一特点可以减少程序员的工作量。

最后附上,我遇到过的函数段:

#define MAX_CALCULATE_COUNT 		   (48)
static int battery_core_calculate_average_voltage(struct battery_core *bat, int cur_voltage)
{

#define BATTERY_VOLTAGE_ERROR_UP       (25 * 1000)
#define BATTERY_VOLTAGE_ERROR_DOWN     (-50 * 1000)

	static int vol_pre = 0;
	static int vol_error = 0;
	static int vol_recover = BATTERY_VOLTAGE_ERROR_UP;
	static unsigned char cur_index = 0;
	static unsigned char max_index = 0;
	static int vol_ma[MAX_CALCULATE_COUNT];
	int average;
  	static int pre_average = 0;
	int dv = 0;
	int i;

	if(!bat->props.present) {	
		vol_pre = 0;
		vol_error = 0;
		vol_recover = BATTERY_VOLTAGE_ERROR_UP;
		cur_index = 0;
		max_index = 0;
		return cur_voltage;
	}

	/*voltage check and auto-correct*/
#if 1
	do {
		if(vol_pre > 0) {
			dv = cur_voltage - vol_pre;
			if(dv <= BATTERY_VOLTAGE_ERROR_DOWN) { /*We assume the battery output big current!*/
				vol_error =   (dv * 9) / 10;
				vol_recover =  - dv / 2;
			} else if(dv >= vol_recover ) { /*We assume the battery recovery from big current output!*/
				vol_error = 0;
				vol_recover = BATTERY_VOLTAGE_ERROR_UP;
			} else {
				vol_error = 0;
				vol_recover = BATTERY_VOLTAGE_ERROR_UP;
			}
		}
		vol_pre = cur_voltage;
		cur_voltage -= vol_error;
	} while(0);
#endif
	pr_err("[jason]vol_pre:%d, cur_voltage:%d\n", vol_pre, cur_voltage);//add by jason

	vol_ma[cur_index] = cur_voltage;
	cur_index ++;
	if(cur_index >= ARRAY_SIZE(vol_ma))
		cur_index = 0;
	if(max_index < ARRAY_SIZE(vol_ma))
		max_index ++;

	average = 0;
	for(i = 0; i < max_index; i++) {
		average += vol_ma[i] ;
	}
	average = average / max_index;


	if(bat->params.debug_mode)
		pr_err(" vol_pre = %dmV - (dv=%d, ev=%d)-> %dmV, avg = %dmV\n", 
			vol_pre/1000, dv/1000, vol_error/1000, cur_voltage/1000, average/1000);

	if( bat->props.state == POWER_SUPPLY_STATUS_DISCHARGING && pre_average > 0 && pre_average < average ) {
		return pre_average;
	}

	pre_average = average;
 
	return average;
}

以上函数进行的是平均电压的计算。函数中大量使用static的类型的变量,这样可以保留48组数据进行计算平均电压。

本文部分参考https://www.cnblogs.com/zzm1/p/10690465.html 

  • 2
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值