【问题标题】:difference of `char * str = "asdf"` and `char str[] = "asdf"` , strtok not working [duplicate]`char * str = "asdf"` 和 `char str[] = "asdf"` 的区别,strtok 不工作 [重复]
【发布时间】:2022-01-30 16:53:37
【问题描述】:

我的代码显示了问题 这行得通

   char str[] = "asdf=1=2=3"; // this works
   printf("type:  %s\n", typename(str)); // prints 'pointer to char'
   char *token = strtok(str, "=");
   printf("%s\n", token);

这行不通,为什么?

   char *str2 = "asdf=1=2=3"; // this wont work
   printf("type:  %s\n", typename(str2)); // prints 'pointer to char'
   printf("%s\n", str2);
   char *token2 = strtok(str2, "="); // segmentation fault
   printf("%s", token2); 

编辑:这是类型名宏

#define typename(x) _Generic((x),        /* Get the name of a type */             \
                                                                                  \
        _Bool: "_Bool",                  unsigned char: "unsigned char",          \
         char: "char",                     signed char: "signed char",            \
    short int: "short int",         unsigned short int: "unsigned short int",     \
          int: "int",                     unsigned int: "unsigned int",           \
     long int: "long int",           unsigned long int: "unsigned long int",      \
long long int: "long long int", unsigned long long int: "unsigned long long int", \
        float: "float",                         double: "double",                 \
  long double: "long double",                   char *: "pointer to char",        \
       void *: "pointer to void",                int *: "pointer to int",         \
      default: "other")

【问题讨论】:

  • 您没有发布 typename 宏或函数...应该修改它以区分 char 数组指向 char 的指针, str 不是。

标签: c string char string-literals chararray


【解决方案1】:

原因如下

   char * pointer_to_readonly_memory_string = "asdf";
   char pointer_to_memory_on_the_stack[] = "asdf";

   // pointer_to_readonly_memory_string[0] = 'b'; // segmentation fault
   pointer_to_memory_on_the_stack[0] = 'b';

   printf(" pointer_to_readonly_memory_string: %s\n", pointer_to_readonly_memory_string);
   printf(" pointer_to_memory_on_the_stack: %s\n", pointer_to_memory_on_the_stack);
char * varname = "asdf";

当您使用星号* 初始化字符串并使用= 赋值时,双引号中的字符串"asdf" 会将字符串文字"asdf" 放在内存的只读部分中,并将varname 设为指向它的指针,这使得对该内存的任何写入操作都是非法的。

char * varname = "asdf";

当您使用方括号 [] 并使用 = 分配值时,它会将文字字符串放入只读内存中,并将字符串复制到堆栈上新分配的内存中。这意味着内存将是可修改的。所以你可以这样做:varname[0] = 'b';

【讨论】:

  • 旁注:编译器确实应该警告指针不是const
猜你喜欢
  • 1970-01-01
  • 2011-04-21
  • 2011-02-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2018-06-27
  • 1970-01-01
  • 2015-10-06
相关资源
最近更新 更多