先上代码
for(int i = 0;i<10;++i){
int temp;
}
问题
for循环内部局部变量temp的生存周期是从何时开始到何时结束?
这个问题其实有些复杂,通常我们会认为temp只存在于单次循环中,即在单次循环开始时变量会被创建,循环结束时变量会被销毁。事实上这完全取决于编译器,在gcc 11.3.0中这10次循环的temp变量的地址是完全相同的。
实验
实验代码
int main(){
int *p = NULL;
for(int i = 0; i < 10;++i){
int temp;
if(i%2)temp=i;
printf("i = %d temp address = %x temp = %d\n",i,&temp,temp);
}
printf("p=%x *p=%d \n",p,*p);
}
实验结果
实验结果中显示10循环内部temp的地址完全相同,而且i为偶数时的循环能访问到i为奇数时temp的值,循环结束后仍然能通过temp的地址取出temp的值。
那么temp在单次循环结束后那段内存被释放了吗?答案是没有
通过将上述代码编译成汇编代码后可以发现,不同循环的temp的值存放在栈的同一位置,循环结束后没有进行任何pop操作,也就是说循环结束并没有销毁temp。只不过循环结束后无法通过temp变量名来访问变量(这是C语言语法限制),但是仍然可以根据变量的地址访问到变量的值。
注意:循环与循环之间的变量temp是共用同一块内存的,也就是说不同循环间temp的地址是相同的,因此循环内部有异步(非阻塞)操作时(比如向线程池添加任务时),应特别注意循环内部局部变量共享同一段内存空间这一情况