char *s 和 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]

  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
很抱歉,作为人工智能,我无法直接执行编程任务。但我可以提供思路和代码示例供参考。 思路: 1. 通过循环输入字符串,并将其存储在一个二维字符数组中; 2. 调用函数 find(),查找数组中最小的字符串,并将其下标存入指针所指内存单元; 3. 输出找到的最小字符串。 代码示例: ```c #include <stdio.h> #include <string.h> #define MAX_STR_LEN 6 #define MAX_STR_NUM 100 void find(char* name[][MAX_STR_LEN + 1], int n, int* p) { int i, min_index; char min_str[MAX_STR_LEN + 1]; strcpy(min_str, name[0][0]); min_index = 0; for (i = 1; i < n; i++) { if (strcmp(name[i][0], min_str) < 0) { strcpy(min_str, name[i][0]); min_index = i; } } *p = min_index; } int main() { char name[MAX_STR_NUM][MAX_STR_LEN + 1]; char* name_ptr[MAX_STR_NUM][MAX_STR_LEN + 1]; int i, n, min_index; printf("请输入多个字符串,以####°作为结束标记:\n"); for (i = 0; i < MAX_STR_NUM; i++) { scanf("%s", name[i]); if (strcmp(name[i], "####°") == 0) { break; } name_ptr[i][0] = name[i]; } n = i; find(name_ptr, n, &min_index); printf("最小的字符串为:%s\n", name[min_index]); return 0; } ``` 以上代码中,使用了一个二维字符指针数组 name_ptr,将输入的字符串的指针存储在其中,以便调用函数 find() 查找最小字符串。函数 find() 中,首先设定一个最小字符串 min_str,将数组中第一个字符串赋值给它。然后遍历数组,依次比较字符串,如果发现比 min_str 更小的字符串,就将其更新为新的最小字符串,并记录其下标 min_index。最后将找到的最小字符串的下标存储在指针 p 所指的内存单元中。在主函数中,通过循环输入字符串,直到输入结束标记为止,并调用函数 find() 查找最小字符串,输出结果。 注意,本代码示例中使用了 C 语言的指针和字符串处理函数,如果您不熟悉这些语言特性,建议先学习相关的知识再尝试理解和运行代码。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值