char *s 和 char s[] 的区别小结

本文详细解析了C语言中字符数组chars[]与字符指针char*s的区别,包括它们在内存模型上的不同表现、赋值过程中的注意事项以及如何正确处理两者之间的转换。

最近的项目中有不少c的程序,在与项目新成员的交流中发现,普遍对于char *s1 和 char s2[] 认识有误区(认为无区别),导致有时出现“难以理解”的错误。一时也不能说得很明白,网上也搜了一下相关文章发现一些写的比较好的,综合了一下当教育资料备用。

 

char *s1 = "hello";
char s2[] = "hello";

 

【区别所在】

char *s1 的s1,而指针是指向一块内存区域,它指向的内存区域的大小可以随时改变,而且当指针指向常量字符串时,它的内容是不可以被修改的,否则在运行时会报错。
char s2[]的s2 是数组对应着一块内存区域,其地址和容量在生命期里不会改变,只有数组的内容可以改变

 

【内存模型】
       +-----+     +---+---+---+---+---+---+
   s1: |  *======> | h | e | l | l | o |\0 |
       +-----+     +---+---+---+---+---+---+
       +---+---+---+---+---+---+
   s2: | h | e | l | l | o |\0 |
       +---+---+---+---+---+---+

 

场景一)
char *s1 = "hello";
char s2[] = "hello";
s2=s1;  //编译ERROR
s1=s2;  //OK

 

分析:s2其地址和容量在生命期里不能改变

 

场景二)
char s2[] = "hello";
char *s1 = s2;  //编译器做了隐式的转换 实际为&s2

char *s1 = &s2;

 

分析:以上两个指针复值完全等价,由于编译器会做这个隐式转换也容易导致初学者误认为 char *s 与char s[]是一回事。
      另用第二种在一些编译器甚至会报警告信息。

 

场景三)
char *s1 = "hello";
char s2[] = "hello";
s1[0]='a';  //×运行ERROR( 这一句好像在一些的编译器不会出错,原因待查)
s2[0]='a';  //OK

 

分析:运行时会报错,原因在于企图改变s1的内容,由于s1指向的是常量字符串,其内容是不可修改的,因此在运行时不会通过。而s2指向的是变量区字符串,可以修改。

 

场景四)
让我们来给一个指针的指针赋值,在使用某些含char**参数的函数时会用到,场景二的增强版。
    char *s1="hello";
    char s2[]="hello";
    char *s3=s2;       //★注意这句必须要★
    char **s4=&s3;   //s2(char[])要用两步才能完成赋值
    char **s5=&s1;   //s1(char*) 只需一步
    printf("s4=[%s]\n",*s4);//打印结果:s4=[hello]
    printf("s5=[%s]\n",*s5);//打印结果:s5=[hello]

 

分析:这个例子应当说最能反映出char *与char []的差异,但是由于使用场合不多,新人尤其需要注意。

 

下面是一些char *s1 和 char s2[]相同的地方(同样编译器对char[]做了隐式变化):
1)作为形参完全相同
如:
   void function(char *s1);
   void function(char s1[]);

 

2)只读取不修改的时候
如:
    char *s1="hello";
    char s2[]="hello";
    printf("s1[1]=[%c]\n",s1[1]);   //s1[1]=[e] 
    printf("s2[1]=[%c]\n",s2[1]);   //s2[1]=[e] 
    printf("s1=[%s]\n",s1);         //s1=[hello]
    printf("s2=[%s]\n",s2);         //s2=[hello]

转载地址:http://duanhengbin.iteye.com/blog/1706635
#ifndef CONFIG_PARSER_H #define CONFIG_PARSER_H #include <stdio.h> #include <stdlib.h> #include <string.h> #include <ctype.h> // 配置行节点 typedef struct ConfigLine { char line[512]; struct ConfigLine *next; } ConfigLine; // 配置段节点 typedef struct ConfigSection { char name[64]; // 段名,如 "network" char key[128]; // 键,如 "my_ssid" ConfigLine *lines; // 该段下的所有行 struct ConfigSection *next; } ConfigSection; // 配置对象 typedef struct { ConfigSection *sections; } Config; // 创建配置对象 Config *config_new(); // 释放配置对象 void config_free(Config *config); // 从 hostapd.conf 读取配置(扁平格式,支持多个 bss=) int config_read(Config *config, const char *filename); // 写入 hostapd.conf 格式配置 int config_write(Config *config, const char *filename); // 从 wpa_supplicant.conf 读取配置(network={...} 格式) int config_read_wpa(Config *config, const char *filename); // 写入 wpa_supplicant.conf 格式配置 int config_write_wpa(Config *config, const char *filename); // 获取指定段的值 int config_get_value(Config *config, const char *section, const char *key, const char *name, char *value, size_t size); // 设置指定段的值 int config_set_value(Config *config, const char *section, const char *key, const char *name, const char *value); // 添加新段 int config_add_section(Config *config, const char *section, const char *key); // 删除段 int config_remove_section(Config *config, const char *section, const char *key); // 删除键 int config_remove_key(Config *config, const char *section, const char *key, const char *name); #endif // CONFIG_PARSER_H 这是当前代码,依据修改根据你提供的 gdb 调用栈信息,程序崩溃在 strcmp() 调用中,而调用栈显示: #1 0x00005555555559de in find_section (config=0x55555555a2a0, section=0x55555555705c "General", key=0x0) at config_parser.c:58 说明你在 find_section() 函数中调用了 strcmp(),并且传入了 key = NULL。 🔍 问题定位 从调用栈来看: 在 main.c 第 26 行调用了 config_set_value(...); 该函数又调用了 find_section(config, "General", NULL); find_section() 内部使用了 strcmp(sec->name, key); 因为 key == NULL,导致 strcmp() 崩溃。 🛠 修复方案 ✅ 1. 修复 find_section() 中的 key == NULL 判断逻辑 假设你的 find_section() 函数如下(简化版): ❌ 问题:key == NULL 时调用 strcmp(sec->name, key) 导致段错误 ✅ 修复后的 find_section() 函数 ✅ 同时修复 config_set_value() 中的逻辑(如果适用) ✅ 修复总结 问题 修复方式 find_section() 中 key == NULL 导致 strcmp() 崩溃 加入 key == NULL 判断 config_set_value() 未检查 key 是否为 NULL 增加 NULL 检查 ConfigSection 字段未初始化为 NULL 确保 config_section_new() 初始化 name type config_new() 未初始化 sections 设置为 NULL
08-27
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值