const定义的常量在超出其作用域之后其空间会被释放,而static定义的静态常量在函数执行后不会释放其存储空间。

我们先来说static.

 

static主要有三个作用:

1.修饰局部变量,成为静态局部变量

2.修饰全局变量,成为静态全局变量

3.修饰函数,成为静态函数

我们一个一个来解释.

1.修饰局部变量。成为静态局部变量

我们先来看下面这一段程序:

#include<stdio.h>
void test()
{
	int a = 5;
	a++;
	printf("%d", a);
}
int main()
{
	int i = 0;
	while (i < 10)
	{
		test();
		i++;
	}
}

输出结果是多少呢?

我们看到主函数一个循环是循环10次test函数,然后每执行一次test,都会打印一次a,a从6开始每次都+1(因为a++在前面,所以从6开始).

所以结果应该是6 7 8 9 10 11 12 13 14 15

那结果是不是如我们所想的呢?

我们来看一下运行结果:

这...?好像不符合我们的猜想诶。

为什么会出现10个6呢?

我们仔细想一下,便可以大致知道问题所在。

就是每一次进入循环的时候a都被重置了,所以不会一直加下去。

这是为什么呢?

因为a是我们所创建的一个临时变量,它会随着函数的结束而被销毁,下次再进来时,会被重新创建。

所以会造成我们刚才那种结果。

那我们就要输出那样6-15这样的结果该怎么做呢?

我们可以在test函数里面int前面加上一个static

这样可以让这个局部变量随着函数的结束而不会被销毁,原理稍后会解释的

改成如下:

void test()
{
	static int a = 5;
	a++;
	printf("%d ", a);
}

这样我们再运行一次,来看看结果吧

符合我们的猜想了。

那具体为什么是这样呢?这里就要说到数据在内存中存储的三个区域

栈区   堆区  静态区

栈区:存放局部变量和函数参数等的地方。栈区的作用范围过了之后会自动回收栈区分配的内存,不需要手动管理。

堆区:由比如malloc,realloc等函数所主动申请的内存,使用完之后须用free释放内存,若申请完之后忘记释放内存,则很容易造成内存泄漏。

静态区:静态变量和全局变量所存储的区域,一旦静态区的内存被分配,直到程序全部结束之后才会被释放。

静态区这个换而言之:静态区的生命周期和程序的生命周期是一样的,出了作用范围不会被销毁(上面static int a =5)这个例子,相当于作用范围不变,但生命周期延长了。

那么上面那个例子我们也可以进行解释原理了:

static修饰局部变量时,实际改变的是变量的存储位置,原来在栈区,被修饰后放在了静态区。

所以说除了作用范围之后不会被销毁。

2.static修饰全局变量

现在vs编译器里面创建两个源文件:test1.cpp,test2.cpp

我们首先在文件test1.cpp里面创建一个全局变量,如下:

 

 紧接着,我们知道引用别的文件里的全局变量需要extern这个函数,所以我们再test2.cpp这个文件里引用一下这个全局变量。

下面看输出结果:

 

 完美的输出了外部的文件。

但是如果我们再test1.cpp里面的全局变量前面加上static,我们可以看一下程序能否成功运行。

我们运行这段程序:

 

程序出错了.

我们看错误原因:无法解析的外部符号。

这是为什么啊,我们明明已经用了extern这个函数来调用外部文件里的那个全局变量了啊,为什么无法解析呢?

 

 那肯定是static捣的鬼嘛,它在这里发挥了什么作用呢?

一个全局变量本来是具有外部属性的,但是被static修饰后,外部属性就变成了内部连接属性,只能在自己所在的源文件内使用,不能在其他文件内使用。

这样呢,被static修饰后的全局变量给我们的感觉是作用域变小了(只能在自己文件内部使用,不能在别的文件内使用),但实质是链接属性变了。

3.static修饰函数

这个和那个修饰全局变量的类似。

还是创建两个源文件test1.c、test2.c。(此时文件名后缀为.c,不是.cpp,在c++环境中无法看到这种效果)

我们在test1.cpp文件里写如下的段程序:

 

 紧接着我们在test2.c里面写Add函数内容。

 

结果:

 完美的输出了我们想要的结果。

但如果我们在test2.c里面的Add函数里面加上static:

再次运行 :

发生错误。

错误原因也是无法解析的外部符号。

错误呢,和上个修饰全局变量一样:

 一个函数本来是具有外部连接属性的,但是被static修饰后,外部链接属性就变成了内部连接属性,只能在自己所在的源文件内部使用,不能在其他文件内部使用了。

再来说一下const

如果把const放在变量类型名前,说明这个变量的值是保持不变的,该变量必须在定义时初始化,初始化后对它进行的任何赋值都是非法的。

1.const修饰常变量
例如:

int a=5;

a=6;

此时a的值被修改为6。

const int a=5;

 此时变量a的值便可再被修改

若仍写a=6,则程序便会出现错误。

2.const修饰常量静态字符串

例如:

const char* str="fdsafdsa";

如果没有const的修饰,我们可能会在后面有意无意的写str[4]=’x’这样的语句,这样会导致对只读内存区域的赋值,然后程序会立刻异常终止。有了const,这个错误就能在程序被编译的时候就立即检查出来,这就是const的好处。让逻辑错误在编译期被发现。

3.修饰函数的参数

直接举例子说明:

1.防止指针修改指针指向的内容内容

我们写以下函数

void String(char* str1,const char* arr2);

这个时候我们可以在String这个函数里面修改arr1内容,但是如果修改arr2程序便会报错,此时arr2不可再被修改。

或:

2.防止指针修改指针指向的地址

void Swap(int* const p1,int* const p2);

此时p1和p2指向的地址便不可再被修改。

以上就是static和const的大致区别,当然会有很多的遗漏之处,欢迎补充!

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐