5

使用 c 风格的字符串,如何将 char 分配给字符指针指向的内存地址?例如,在下面的示例中,我想将 num 更改为“123456”,因此我尝试将 p 设置为 '0' 所在的数字,并尝试用 '4' 覆盖它。谢谢。

#include <stdio.h>
#include <stdlib.h>

int main()
{
    char* num = (char*)malloc(100);
    char* p = num;

    num = "123056";

    p = p+3;    //set pointer to where '4' should be
    p = '4';

    printf("%s\n", num );

    return 0;
}
4

5 回答 5

13

首先,当你这样做时:

num = "123056";

您没有将字符串“123056”复制到由malloc(). 在 C 中,为指针分配一个char *字符串字面值相当于将其设置为一个常量 - 即等同于:

char str[] = "123056";

所以,你刚刚完成的是你放弃了对 100 字节堆区域的唯一引用malloc(),这就是为什么你的后续代码没有打印正确的值;' p' 仍然指向malloc()(因为num在分配时指向它)分配的堆区域,但num不再指向。

我假设您实际上打算将字符串“123056”复制到该堆区域中。以下是如何做到这一点:

strcpy(num, "123056");

虽然,出于多种原因,这是更好的做法:

strncpy(num, "123056", 100 - 1);  /* leave room for \0 (null) terminator */

如果您刚刚完成:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int     main(void) {
        char    *num = malloc(100);
        char    *p = num;

        strncpy(num, "123056", 100 - 1);

        p = p + 3;
        *p = '4';

        printf("%s\n", num);

       return 0;
} 

你会得到正确的结果:

123456

您可以收缩此操作:

p = p + 3;
*p = '4';

...并通过如下方式避免迭代指针:

*(p + 3) = '4';

其他一些注意事项:

  • malloc()尽管常见的风格实践,强制转换to的返回值(char *)是不必要的。类型的转换和对齐void *由 C 语言保证。

  • 总是检查的返回值malloc()。如果堆分配失败(即您的内存不足),它将为 NULL,此时您的程序应该退出。

  • 根据实现,分配的内存区域malloc()在某些情况下可能包含陈旧垃圾。在分配后将其归零总是一个好主意:

    memset(num, 0, 100);
    
  • 永远不要忘记free()你的堆!在这种情况下,程序会退出,操作系统会清理你的垃圾,但如果你不养成这种习惯,你很快就会发生内存泄漏。

因此,这是“最佳实践”版本:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>

int     main(void) {
        char    *num, *p;

        /*
         * Don't take 1-byte chars for granted - good habit to get into.
         */

        num = malloc(sizeof(char) * 100);

        if(num == NULL)
                exit(1);

        memset(num, 0, sizeof(char) * 100);

        p = num;

        strncpy(num, "123056", 100 - 1);

        *(p + 3) = '4';

        printf("%s\n", num);

        free(num);

        return 0;
}
于 2009-06-03T05:35:09.303 回答
12

该代码不起作用,仅仅是因为该行:

num = "123056";

更改num为指向远离分配的内存(并且p仍然指向该内存,因此它们不再是相同的位置)到最有可能的只读内存。您不允许更改属于字符串文字的内存,这是未定义的行为。

您需要以下内容:

#include <stdio.h>
#include <stdlib.h>
int main (void) {
    char *num = malloc (100); // do not cast malloc return value.
    char *p = num;

    strcpy (num, "123056");   // populate existing block with string.

    p = p + 3;                // set pointer to where '0' is.
    *p = '4';                 // and change it to '4'.

    printf ("%s\n", num );    // output it.

    return 0;
}
于 2009-06-03T05:40:11.573 回答
1

除了其他人指出的 *p 问题外,您还有内存使用问题。您有一个 100 字节的缓冲区,内容未知。您还有另一个 7 字节的缓冲区,其中包含字符串“123056”和一个空终止符。你这样做:

  1. num 设置为指向 100 字节的缓冲区
  2. p 设置为指向 num;即,它指向 100 字节的缓冲区
  3. 您将 num 重置为指向 7 字节缓冲区;p 仍然指向 100 字节的缓冲区
  4. 您使用 p 修改 100 字节缓冲区
  5. 然后您使用 num 打印出 7 字节缓冲区

因此,您打印的缓冲区与您正在修改的缓冲区不同。

于 2009-06-03T05:38:37.457 回答
0

只需使用*p = '4';...!

于 2009-06-03T05:32:04.193 回答
0

好吧,你知道 p 是指针类型。它存储 char '0' 的地址。如果您为 p 分配“4”的值。它将以“4”作为地址。但是,地址“4”是非法的。您可以使用 '*' 运算符来获取存储在 p 中的地址的值 stored 。

于 2009-06-03T05:36:53.453 回答