我们都知道指针数组大概可以理解为长这样:(假设char* a[10])
所以我对数组元素的赋值思路是定义一个变量i=0,给一个a[i]赋值后,再将i+1,继续赋值下一个。
针对例题:
(HITOJ测试题)
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
int main()
{
char* p[150];
int n;
printf("How many countries?");
scanf("%d", &n);
getchar();
printf("Input their names:\n");
for (int i = 0;i < n;i++)
{
char str[] = " ";
p[i] = (char*)malloc(sizeof(str));
gets(str);
p[i] = str;
}
for (int i = 0;i < n;i++)
{
puts(p[i]);
}
return 0;
}
运行程序后发现:
问题1:为什么str的栈溢出了?
问题2:为什么明明给p[0]赋的是China的地址,p[1]赋的是America的地址,而打印出的p[0]内容也为America?
针对问题1,因为在
for (int i = 0;i < n;i++)
{
char str[] = " ";
p[i] = (char*)malloc(sizeof(str));
gets(str);
p[i] = str;
}
这个for循环中,char str[]的大小会被第一次的输入的内容所确定,即str[5],而America超出了这个范围。
针对问题2,是因为char str[]在这个for循环里不会被释放,每次for循环进行时str可以被访问,但不会重新建立。而p[i]=str,而不是直接被赋值为键入的字符串。所以str变了,p[i]都得变。画个图来加强理解:(此时p指针数组的情况)
解决方案:
针对问题1,可以将str提前规定大小,比如str[50],防止栈溢出;(后面好像也不需要str了orz)
针对问题2,我们直接给p[i]赋值就OK,不用通过str[]中转复制内容。
解决代码如下:
#define _CRT_SECURE_NO_WARNINGS 1
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
int main()
{
char* p[150];
int n;
printf("How many countries?");
scanf("%d", &n);
getchar();
printf("Input their names:\n");
for (int i = 0;i < n;i++)
{
p[i] = (char*)malloc(sizeof(char));
gets(p[i]);
}
for (int i = 0;i < n;i++)
{
puts(p[i]);
}
return 0;
}
就在我写博客的时候,突然想到p[i]的大小就是char*的大小,也就是1个字节,为什么还要malloc一下给他一个char大小的空间呢?
于是我就把
p[i] = (char*)malloc(sizeof(char));
这句话注释了,发现程序不能正常进行了 ,输入完China敲下回车后程序就自动结束了。
why?就作为我下次博客的内容吧