C语言中*号和[]的区别

char s[] = "My name is XXXXXX.";

memmove(s, s + 11, strlen(s) + 1 - 11);

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

这段代码是正常的,但是把s[]换成*s后,执行就报错了,请教其区别

阅读 5.2k
5 个回答

不纯是语言问题, 和编译器也有关

char s[] = "xxx";是用string literal来初始化一个栈上的数组, 得到的s[]是在栈上 (运行时可写的页), 初始化后的s已经和string literal没关系了.

char *s把s指向那个string literal本身, 很可能是不可写的内存. 比如linux的gcc会把char *s指向的字符串放到elf的.rodata区 (运行时被载入到不可写的页). 试图朝不可写的页memmove会segfault.

#include <stdio.h>
int main(int argc, char *argv[])
{
    int a = 1;
    char *s1 = "My name is XXXXXX.";
    char s2[] = "My name is XXXXXX.";
    int b = 1;

    printf("%p\n", &a);
    printf("%p\n", s1);
    printf("%p\n", s2);
    printf("%p\n", &b);
    return 0;
}

另外"修改string literal所在内存"是未定义行为, 知道原理就可以了, 不要这样做

char *s是字符指针,存储容量就是一个指针(sizeof(char *)

char s[]是字符数组,存储容量为后面的字符数量(sizeof(char)*数量
memmove需要有至少等于字符容量的空间才能复制字符串

新手上路,请多包涵

char s只是一个指针,4个字节,需要申请空间s=(char )malloc ()

一句话 char *定义的变量是不可以修改其内容的

新手上路,请多包涵

char* s = "xxxxx" 实质是在全局变量区分配了一个字符串存储空间,并将这个空间的首地址赋值给s(编译器过程中执行);在C语言中对于堆区中字符串常量是不允许修改的,所以在调用memmove对s内容修改时,产生错误提示。

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进