单引号和双引号

  • C 语言中的单引号用来表示字符字面量
  • C 语言中的双引号用来表示字符串字面量
'a' 表示字符字面量,在内存中占用 1 个字节,'a' + 1 表示 ASCII码加 1 ,结果为 b

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

实例分析:单引号和双引号的本质

#include <stdio.h>

int main()
{
    char* p1 = 1;        // warning: initialization makes pointer from integer without a cast

    char* p2 = '1';      // warning: initialization makes pointer from integer without a cast

    char* p3 = "1";
    
    printf("%s", p1);    // 段错误 
    printf("%s", p2);    // 段错误
    printf("%s", p3);
    
    printf('\n');        // 段错误 ; warning: passing argument 1 of ‘printf’ makes pointer from integer without a cast
    printf("\n");
    
    return 0;
}

小贴士

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

程序实例分析 1

clipboard.png

char* p1 = 1; 将立即数 1 赋值给 p1 ,p1 指针便指向 0x00000001
char* p2 = 1; 将立即数 31 赋值给 p2, p2 指针便指向 0x00000031
char* p3 = "1"; 编译 "1" 时,编译器会分配合法内存空间, p3 便指向 "1" 的内存地址

程序实例分析 2

clipboard.png
'\n' printf 将访问地址为立即数 0x00000010 地址
"\n" 编译 "\n" 时,编译器会分配合法内存空间, printf 便访问 "\n" 的内存地址

实例分析: 混淆概念的代码

#include <stdio.h>

int main()
{
    char c = " ";    // warning: initialization makes pointer from integer without a cast
                     // 将 4 字节内存地址赋值1字节类型,将发生数值截断 
    
    while( (c == "\t") || (c == " ") || (c == "\n") )
    {                // warning: comparison between pointer and integer
        scanf("%c", &c);s
    }
    
    return 0;
}
输出:
非期望运行结果!

程序实例分析

char c = "string"; 发生了什么?

分析:【实际发生的是 ASCII 与 地址的比价】

  • 编译后字符串 " string" 的内存地址被赋值给变量 c
  • 地址占用 4 个字节,而变量 c 占用一个字节
  • 由于类型不同,赋值产生截断

小结

  • 单引号括起来单个字符代表整数
  • 双引号括起来的字符代表指针
  • C 编译器接收字符和字符串的比较,无任何意义【编译器发出警告】
  • C 编译器允许字符串对字符变量赋值,只能得到错误 【编译器发出警告】

以上内容参考狄泰软件学院系列课程,请大家保护原创!


TianSong
734 声望138 粉丝

阿里山神木的种子在3000年前已经埋下,今天不过是看到当年注定的结果,为了未来的自己,今天就埋下一颗好种子吧