今天写OJ时,发现了个小问题,记录一下。方便后期查询!
OJ题:https://www.acoj.com/problems/10004

  • 相关思路:
#include <stdio.h>

int main(){
	int b[1005]; 
	int N,i,t,num=0;
	scanf("%d",&N);
	for(i=1;i<=N;i++){
		scanf("%d",&t);
		if(b[t]==0){	// 接受数据,并标记为1,表示其出现过,计入结果 
			b[t]=1,num++;
		}
	}
	printf("%d\n",num);
	for(i=1;i<=1000;i++){
		if(b[i]==1){
			printf("%d ",i);
		}	
	}
	return 0;
} 

乍一看,没毛病。但一运行:


然后就疯狂地Debug,结局竟然只需要:

对!将数组移到外面!!!!可是,why???

  • 参考网络:

  • 先介绍一下栈区(stack),堆区(heap),数据区(data seg)和代码区。

    • 栈区:由操作系统自动分配释放,存放函数的参数值,局部变量的值;当不需要时系统会自动清除。
    • 堆区:由new分配的内存块,不由编译器管,由应用程序控制(相当于程序员控制)。如果程序员没有释放掉,程序结束后,操作系统会自动回收。
    • 数据区:也称全局区或者静态区,存放全局的东西类似全局变量。
    • 代码区:存放执行代码的地方,类似if else,while,for这种语句。
  • <mark>在main函数里面的数组是开在栈区(stack),在函数外面的是开在数据区的。栈区的内存比较小,所以当数组非常大的时候,就会报错。假如把数组放在数据区就不会出现这个问题,因为数据区的内存很大</mark>。

  • 也就是说,在main函数外面开一个数组,他的内存分配在数据区里;如果在main函数内部开数组,内存分配在栈区内。一般来说栈区的内存是比较小的,所以平常开一些小一点的数组是没问题的;但如果题目要求的数组比较大,那就会出现爆出的问题,程序无法访问内存就会出错;相对的,数据区的内存较大,<mark>所以开数组开在数据区/main函数外面,就不易出现这样的问题</mark>。

  • 这也是为什么很多博客和OJ题解将数组开到主函数外面!!!