【C语言进阶剖析】14.C语言中的单引号和双引号

文章目录

一、单引号和双引号

二、小贴士

三、程序实例分析1

四、程序实例分析2

五、容易混淆的代码

六、小结


一、单引号和双引号

  • C语言中的单引号用来表示字符字面量
  • C语言中的双引号用来表示字符串字面量

        'a'表示字符字面量,在内存中占1个字节,'a'+1表示'a'的ASCII码加1,结果为'b'

"a"表示字符串字面量,在内存中占2个字节,"a"+1表示指针运算,结果指向"a"结束符'\0'

        下面看一段单引号和双引号本质的代码:

#include <stdio.h>

int main()
{

    char* p1 =  1 ;
    char* p2 = '1';
    char* p3 = "1";

    printf("%s, %s, %s", p1, p2, p3);
    
    printf('\n');
    printf("\n");
    
    return 0;
}

        编译后会有警告,继续运行会发生段错误,如下:

         究竟问题出现在哪里呢?下面先把打印语句注释掉并运行:

#include <stdio.h>

int main()
{

    char* p1 =  1 ;
    char* p2 = '1';
    char* p3 = "1";

    //printf("%s, %s, %s", p1, p2, p3);
    
    //printf('\n');
    //printf("\n");
    
    return 0;
}

        编译运行虽然有警告,但是没有段错误,那么段错误肯定是打印语句那里了

         把打印语句去掉注释

#include <stdio.h>

int main()
{

    char* p1 =  1 ;
    char* p2 = '1';
    char* p3 = "1";

    printf("%s, %s, %s", p1, p2, p3);
    
    //printf('\n');
    //printf("\n");
    
    return 0;
}

        编译运行后果然出现了段错误

         究竟是怎么回事呢?接着往下看

二、小贴士

  • 字符字面量被编译为对应的 ASCII 码
  • 字符串字面量被编译为对应的内存地址
  • printf 的第一个参数被当成字符串内存地址
  • 内存的低地址空间不能在程序中随意访问

三、程序实例分析1

        注意:字符 '1' 所对应的 ASCII 码就是十六进制的 0x00000031

        0x08048000 这个内存地址是非常特别的内存地址,所有低于这个地址的内存空间是不能随便访问的,一旦访问,就会发生段错误。程序里面的字符串编译之后得到的地址都是大于 0x08048000,所以在程序中可以访问。

         所以说 1 和 '1' 的内存地址都不能访问,访问了就会出现段错误。如果把 printf("%s, %s, %s", p1, p2, p3); 注释打印下面的语句会发生什么?

#include <stdio.h>

int main()
{

    char* p1 =  1 ;
    char* p2 = '1';
    char* p3 = "1";

    //printf("%s, %s, %s", p1, p2, p3);
    
    printf('\n');
    printf("\n");
    
    return 0;
}

        输出结果如下,还是会发生段错误:

四、程序实例分析2

        '\n' 字符字面量还是处于地址值区间,所以访问这个地址就会发生段错误。

         如果把 printf('\n'); 注释掉

#include <stdio.h>

int main()
{

    char* p1 =  1 ;
    char* p2 = '1';
    char* p3 = "1";

    //printf("%s, %s, %s", p1, p2, p3);
    
    //printf('\n');
    printf("\n");
    
    return 0;
}

        编译就不会出错,而且换了一行:

五、容易混淆的代码

        这段代码的本意是如果字符 c 是 "\t" 或者 " " 或者 "\n",,则输入字符。

#include <stdio.h>

int main()
{

    char c = " ";
    
    while( (c == "\t") || (c == " ") || (c == "\n") )
    {
        scanf("%c", &c);
    }
    
    return 0;
}

        编译运行后,发现程序并没有让我们输入:

         在进行解释之前,先看下面这样会发生什么:

         分析:
                1)编译后字符串"string"的内存地址被赋值给变量 c

                2)内存地址占用 4个字节,而变量 c 只占用1个字节

                3)由于类型不同,赋值后产生截断

        所以说 char c = " ";  赋值后产生截断,那么 while 里面的语句也就不会执行了,这段程序是混淆了字符和字符串的概念。

        可以这么改:

#include <stdio.h>

int main()
{

    char c = ' ';
    
    while( (c == '\t') || (c == ' ') || (c == '\n') )
    {
        scanf("%c", &c);
    }
    
    return 0;
}

        这样就能正确了:

六、小结

  • 单引号括起来的单个字符代表整数
  • 双引号括起来的字符代表字符指针
  • 编译器接受字符和字符串的比较,无任何意义
  • 编译器允许字符串对字符变量赋值,只能得到错误
  • 44
    点赞
  • 179
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

清风自在 流水潺潺

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值