本人在字符串指针数组中给元素赋值时的误区(赋值内容重复)

文章讲述了在编程中使用指针数组时遇到的问题,如栈溢出和内存管理。作者分析了问题1中str数组大小导致的溢出,以及问题2中内存分配与字符串直接赋值的误解。给出了修改后的代码和额外思考关于内存分配的疑问。
摘要由CSDN通过智能技术生成

 我们都知道指针数组大概可以理解为长这样:(假设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?就作为我下次博客的内容吧

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值