第2篇:C/C++中的字符字面量

字符字面量

字符常量是单引号引起来的字符序列 ,字符常量通常由一个字符组成 , 也可以包含多个字符,比如'\n',在C中字符字面量是以int类型表示的,因此字符字面量的sizeof运算结果是4。更确切地说,在C中字符字面量是ASCII系统编码对应的索引,所以例如'a'就对应整数37.'A'对应的就是65.可见"C Programming Language"这本书的第35-36页的相关文档.

必须要提到一个概念就是数据对象(Data Objects),这里简称对象,这是很多写C教程很少提到的一个概念,也不是我凭空捏造的.文献来源http://www.newtonlabs.com/ic/ic_5.html
像基于C衍生出来其他高层语言所理解的"对象"是有些区别的.C中的对象更偏向于内存模型,数据对象就是规范C中所有基本数据类型的两个基本属性.

  • 数据值(value):就是存储在某片特定数据类型大小的内存块中二进制数据的表示形式。
  • 存储位置 (storage location),该存储位置由操作系统为分配一个该位置对应的地址

备注:一般来说,数据值对应的是右值,而存储位置对应的是左值,也就是变量

字符字面量是数据对象中的一个数据值,C的编译器根本不允许你这么做,因为数据值只是在某个一字节的内存块中二进制码表示形式,所以不会存在用取址操作符(&)去获取一个值的地址这种荒谬的说法.

char类型和字符字面量

其实这个问题就是上面同一个问题,换了一种问题描述而已。

因为在C中的声明一个char类型的变量就是一个左值(存储位置),它仅仅在char类型的变量对应的内存空间中保存字符字面量对应的ASCII编码的int类型的值而已.例如:

char c='A';

就告诉了C编译器完成两件事:

  1. 需要分配char类型该有的内存空间(通常是8bit,但实际大小跟运行的硬件平台有关);
  2. 在为char变量分配的内存空间中保存ASCII码字面量'A'对应的整数值65(二进制形式)


    字符字面量在内存中的表示

字符字面量:C vs C++

我们来考虑一下两个简单的代码

  • 在C中char是一个基本的数据对象,默认1个字节;
  • 在C++中同样继承了C的char类型,当然默认也是1个字节;
    printf("%ld",sizeof(char));   //1个字节
    

C++中的字符字面量
下面这个代码的输出就非常有意思了

printf("%ld",sizeof('a'));   
  • 在C中,因为'a'本质上就是返回ASCII编码对应的int类型的整数值,而该C中int默认就是4个字节.这个没疑问.
  • 在C ++语言中,当字符字面量被sizeof作为一个操作数时,C++编译器就会在调用代码的上下文隐含地定义了 一个临时char变量,并且该字符字面量对应的编码值保存到该临时变量中.因此你在C++源文件中用sizeof('a')会被替换为 以下伪代码
     char tmp='a';
      sizeof(tmp); //打印char类型变量的尺寸,而不是ASCII码对应int类型的尺寸
    

来源中文翻译:窄字符字面量或原始字符文字,例如 'a'或'\ n'或'\ 13'。 这样的字面量具有char类型,并且值等于执行字符集中c-char表示的值。 如果c-char不能表示为执行字符集中的单个字节,则字面量具有int类型和实现定义的值。
文献来源:https://en.cppreference.com/w/cpp/language/character_literal

其中为什么C++中要字符字面量视为一个char类型来看待其中一个例证那就是C++为了兼顾函数在面向对象编成思想中的两个特性,就是同样函数名称的原型的多态重写的这两个特性,试想以下例子。

char kiss_me(char c); 

int kiss_me(int c);


void main(void){
     kiss_me('A'); 
     kiss_me(65);
}

作一个反证的假设,如果C++将字符串字面量和C的约定是一样的话,那么字符字面量就视为一个ASCII码对应的int类型的整数值,那么C++就无法分辨下面的关于kiss_me的不同函数原型了,这就很矛盾了,反证C++中要字符字面量视为一个char类型的必要性.

区分字符字面量和字符串字面量

字符串字面量即用双引号括着的多个字符字面量就是字符字面量的序列。并且字符串字面量是存在于字面量池中,字面量池中的字符串字面量是在程序初始化的时候填出入字面量池中,其生命周期直到加载该字面串字面量的程序退出。我们可以将字面量池想象成一份“签到表”,而字符串字面量就比喻成众多的签名,但这份签到表比较特殊。

ss17.png

字面量池的特征

字面量池就是位于C/C++中内存布局中较向低地址的一片内存区域,这片内存区域的通常是公开的,整个程序的上下文用到所有字符字面量都会在初始化过程中填入该池,并且全局维护整个程序上下文的唯一副本。也就是其他内存区域用到字符串字面量副本是从该池中“初次”(注意我的用词)拷贝过去的发源地。该池的特征如下。

  • 不允许有重复的字符串字面量出现,这样做的目的是为了节省空间,例如"hello word"不可能出现两次,但“hello world ”,注意后者的字符字面量最后是一个空格即
    char s[]=['h','e','l','l','o',' ','\0'];  
    
    因此被字面量池认为不同于前者的字符串字面量。下面的例子同样的字符串字面量出现了两次,但两个char指针都指向字面量池中同一个字符串字面量,也就是说指向其池中的内存地址是一样的。
     int main (void){
        char *s="it-dog";
        char *b="it-dog";
        //指针s和b指向同一个字符字面量的内存地址
        //0x557375b2b21d
    }
    
  • “字符串字面量”之间都是连贯的,每个字符串字面量之间结尾存在一个特殊字符'\0'
  • 每个字符串字面量都存在一个唯一的内存地址,也就是说前一个字符串字面量末端字符'\0'的下一个字符字面量的内存地址就是对应字符串字面量的内存地址,即&“hello word”这样的用法在C/C++中是合法的。
    char* s="hello word";
    &"hello word" //和上面的
    printf("%lp\n",&s) //和下一条语句输出的内存地址是一样的
    printf("%lp\n",&"hello word")
    
  • 字面量池中的字符串字面量是只读的,但池外中的其他内存区域的变量持有的相同的字符串字面量只是一个相同的副本而已,只要没有声名为常量字符串,都可以修改。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349