首页 C语言标准参考手册

C语言标准参考手册

举报
开通vip

C语言标准参考手册1 C的标准化过程 C语言自诞生到现在,期间经历了多次标准化过程,主要分成以下几个阶段: 1.1 Traditional C 此时的C语言还没有标准化,来自“C Programming Language, First Edition, by Brian W. Kernighan, Dennis M. Ritchie. Prentice Hall PTR 1978”的C描述可算作“正式”的标准,所以此时的C也称为“K&R”C。 期间C语言一直不断的发生细微的变化,各编译器厂商也有自己的扩展,这个过程一直持续到20世纪...

C语言标准参考手册
1 C的 标准 excel标准偏差excel标准偏差函数exl标准差函数国标检验抽样标准表免费下载红头文件格式标准下载 化过程 C语言自诞生到现在,期间经历了多次标准化过程,主要分成以下几个阶段: 1.1 Traditional C 此时的C语言还没有标准化,来自“C Programming Language, First Edition, by Brian W. Kernighan, Dennis M. Ritchie. Prentice Hall PTR 1978”的C描述可算作“正式”的标准,所以此时的C也称为“K&R”C。 期间C语言一直不断的发生细微的变化,各编译器厂商也有自己的扩展,这个过程一直持续到20世纪80年代末。 1.2 C89 考虑到标准化的重要,ANSI(American National Standards Institute)制定了第一个C标准,在1989年被正式采用(American National Standard X3.159-1989),故称为C89,也称为ANSI C。 该标准随后被ISO采纳,成为国际标准(ISO/IEC 9899:1990)。 C89的主要改动: ​ 定义了C标准库; ​ 新的预处理命令和特性; ​ 函数原型(prototype); ​ 新关键字:const、volatile、signed; ​ 宽字符、宽字符串和多字节字符; ​ 转化规则、声明(declaration)、类型检查的改变。 1.3 C95 这是对C89的一个修订和扩充,称为“C89 with Amendment 1”或C95,严格说来并不是一个真正的标准。 C95的主要改动: ​ 3个新标准头文件:iso646.h、wctype.h、wchar.h; ​ 一些新的标记(token)和宏(macro); ​ 一些新的printf/scanf系列函数的 格式 pdf格式笔记格式下载页码格式下载公文格式下载简报格式下载 符; ​ 增加了大量的宽字符和多字节字符函数、常数和类型。 1.4 C99 这是目前最新的标准,由ISO制定于1999年(ISO/IEC 9899:1999),故称为C99。 C99的主要改动: ​ 复数(complex); ​ 整数(integer)类型扩展; ​ 变长数组; ​ Boolean类型; ​ 非 英语 关于好奇心的名言警句英语高中英语词汇下载高中英语词汇 下载英语衡水体下载小学英语关于形容词和副词的题 字符集的更好支持; ​ 浮点类型的更好支持; ​ 提供全部类型的数学函数; ​ C++风格注释(//)。   2 C标准文档 2.1 C99 这是一个pdf文件:c99.pdf。 2.2 C89 C99已经替代C89成为标准,所以C89文档已经很难找了。 下面是书籍“C Programming Language, Second Edition, by Brian W. Kernighan, Dennis M. Ritchie. Prentice Hall PTR (April 1, 1988), 0131103628.”附录A的一份拷贝,在此作为C89标准以供需要时查阅。 同时也提供中文版本,内容来自该书对应的中译版“《C程序 设计 领导形象设计圆作业设计ao工艺污水处理厂设计附属工程施工组织设计清扫机器人结构设计 语言》,徐宝文等译,机械工业出版社出版,ISBN 7111075897”。 文档仅供个人参考使用(建议以英文版为主):         英文版        中文版 为方便起见,提供一份标点符号中英对照表,希望有用:标点符号中英对照表。   3 C标准的选择 选择标准依赖于编译器的支持和对可移植性的要求。 C99是当前的标准,但它仍未得到广泛支持,虽然标准发布已经多年。C99对C89(C95)的改动非常大,如果编写C99的代码,那么可移植性必然受到限制。此外,个人认为C99的一些新特性在大多数程序设计中并不是必须的。 C89(包括C95)是目前使用最广泛的,并得到所有主流编译器的支持。 Traditional C现在只会在一些非常老的代码中才能见到了,除非你在维护旧代码,否则不应该再使用它。 所以,个人觉得当前还是以C89(包括C95)标准为主。 1 引言 本手册描述的c语言是1988年10月31日提交给ANSI的草案,批准号为“美国国家信息系统标准―C程序设计语言,X3.159-1989”。尽管我们已非常小心,以便这个手册的介绍可以信赖,但它毕竞不是标准本身,而是对标准的一个解释。这个手册的安排基本与标准相似,也与本书的第1版相似,但是对细节的组织是不同的。本手册给出的语法与标难是一样的,只是有少量产生式有所修改,词法元素和预处理器的定义也非形式化。注释部分说明了ANSI标准C与本书第1版介绍的或其他编译器所支持的语言的细微差别。     2 词法规则 一个程序由存储在文件中的一个或多个翻译单元织成,程序的翻译分几个阶段完成,这将在12节中介绍。翻译的第一阶段完成低级的词法转换,执行由字符#开始的行所引入的指令,井进行宏定义和宏扩展。当预处理(将在12节中介绍)完成后,程序就被归约成一个单词序列。   2.1 单词 共有6类单词;标识符、关键字、常量、字符串字面值、运算符和其他分隔符。空格、横向和纵向制表符.换行符、换页符和注解(合称空白符)在程序中仅用来分隔单词,因此将被略过。空白符用来分开相邻的标识符、关键字和常量。 如果到某一字符为止的输人流被分成若干单词.那么下一个单词就是可能组成单词的最长的字符串。   2.2 注解 注解以字符/*开始,以*/结束。注解不可以嵌套,也不可以出现在字符串中或字符字面值中。   2.3 标识符 标识符是一个字母和数字的序列,其第一个字符必须是―个字母,下划线_也被当做字母。大写和小写字母组成的标识符是不同的。标识符可以任意长。对于内部标识符,至少前31个字母是有意义的,在某些实现中这个值可以更大。内部标识符包括预处理的宏名和其他没有外部连接(见11.2节)的名字。有外部连接的标识符的限制要多一些,其实现可能只认为前6个字符是有意义的.而且有可能忽略大小写的不同。   2.4 关键字 以下标识符被保留为关键字,它们不能用做别的用途: auto double int struct break else long switch case enum register typedef char extern return union const float short unsigned continue for signed void default goto sizeof volatile do if static while 有些实现还把单词fortran和asm保留为关键宇。 ​ 关健字const、signed和volatile是ANSI标准中新增加的,enum和void是第1版后新增如的,entry曾经被保留为关键字,但现在已不是了。   2.5 常量 共有几种类型的常量,它们每一种都有一个数据类型,基本类型将在4.2节讨论。     常量:       整数常量       字符常量       浮点常量       枚举常量 2.5.1 整数常量 整数常量由一串数字序列组成。如果它以0(数字0)开始,那么是八进制数.否则就是十进制数。八进制常量不包括数字8和9。以0x和0X(数字0)开始的数字序列是十六进制数,十六进制数包含到从a~f或从A~F的字母,它们分别表示10~15。 一个整数常量可以以宁母u或U为后缀,表示它是一个无符号数;也可以以字母l或L为后缀,表示它是一个长整数。 一个整数常量的类型依赖于它的形式、值和后缀(类型的讨论见4节)。如果它没有后缀且是十进制的,那么它的类型很可能是int、long int或unsigned long int。如果它没有后缀且是八进制的或十六进制的,那么它的类型很可能是int、unsigned int、long int、unsigned long int。如果它的后缀为u或U,那么它的类型很可能是unsigned int或unsigned long int。如果它的后缀为l或L,那么它的类型很可能是long int或unsigned long int。 ​ 整数常量类型的确定比第1版要详细得多;在第1版中,大的整数常量仅被看做是long类型的。U后缀是新增加的。 2.5.2 字符常量 字符常量是由单引号括住的一个或多个字符的序列,如'x'。单字符常量的值是执行时机器的字符集中的此字符的数值,多字符常量的值由实现定义。 字符常量不包括字符'和换行符,为了表示它们和某些其他的字符,可以使用以下的转义序列(换码序列): newline NL (LF) \n 换行符 horizontal tab HT \t 横向制表符 vertical tab VT \v 纵向制表符 backspace BS \b 回退符 carriage return CR \r 回车符 formfeed FF \f 换页符 audible alert BEL \a 响铃符 backslash \ \\ 反斜线 question mark ? \? 问号 single quote ' \' 单引号 double quote " \" 双引号 octal number ooo \ooo 八进制数 hex number hh \xhh 十六进制数 转义序列\ooo由反斜杠后跟1、2或3个用来确定对应字符的值的八进制数字组成。一个普通的例子是\0(其后没有数字),它表示字符NUL。转义序列\xhh由反斜杠开始,后跟x,其后是十六进制数字,用来确定对应字符的值。数字的个数没有限制,但如果对应的字符的值超过最大的字符的值,那么该行为是未定义的。对于八进制或十六进制转义字符,如果实现中将类型char看做是有符号的,那么将对字符值进行符号扩展,就好像它被强制转换为char类型一样。如果\后面的字符不是以上所说明的,那么其行为是未定义的。 在C语言的某些实现中,有一个扩展的字符集,扩展的部分不能用char类型表示。在该扩展集中,常量是由一个前导L开始(如;L'x'),叫做宽字符常量。这种常量的类型为wchar_t。这是一个整数类型,定义在标准头文件中。与通常的字符常量一样.可以使用八进制和十六进制的转义序列;但是,如果值超过wchar_t可以表示的范围,那么结果是未定义的。 ​ 某些转义序列是新增加的,特别是十六进制字符的表示。扩展的字符也是新增加的。通常美国和西欧所用的字符集可以用char编码;增加wchar_t的主要意图是为了表示亚洲的语言。 2.5.3 浮点常量 一个浮点常量包含有一个整数部分、一个小数点、一个小数部分、一个e或E,一个可选的有符号整数类型的指数和一个可选的表示类型的后缀(即f、F、l或L)。整数和小数部分均由数字序列组成。可以没有整数部分或小数部分(但不能二者都没有)。小数点部分或者e和指数部分可以没有(但不能二者都没有)。浮点常量的类型由后缀确定,F或f后缀表示它float类型;1或L后缀表明它是long double类型;若没有后缀则是double类型。 ​ 浮点常量的后缀是新增加的。 2.5.4 枚举常量 定义为枚举符的标识符是int类型的常量(见8.4节)。   2.6 字符串字面值 字符串字面值也叫字符毒常量,是由双引号括起来的一个字符序列,如"..."。字符串的类型为“字符数组”,存储类为static(见4节),由给定的字符来初始化。相同的字符串字面值是否看做是不同的取决于具体的实现。如果程序试图改变字符串字面值,那么该行为是未定义的。 我们可以把相邻的字符串字面值连接为一个单一的字符串。任何连接之后,一个空字节\0被加到字符串的后面以使程序在扫描字符串时知道已到达字符串的末尾。字符串字面值不包含换行符和双引号符;但可以用与字符常量相同的转义序列来表示它们。 与字符常量一样,扩展字符集中字符串字面值以前导字符L表示,如L"..."。宽字符的字符串字面值的类型为wchar_t的数组,将普通字符串和宽字符的字符串字面值进行连接是未定义的。 ​ 如下说明都是ANSI C标准中新增加的:字符串字面值不必相区别、紫止修改字符串字面值以及相邻字符串字面值的连接。宽字符的字符串字面值也是新增加的。     3 语法符号 在本手册用到的语法符号中,语法类别由斜体字表示。字面值单词和字符以打字机字体表示。可选类别通常列在不同的行中,但在少数情况下,一长串短的可选项可以表示在一行中,以短语“之一”(one of)标识。任选的终结符或非终结符带有下标“opt”。例如:     表达式opt 表示一个括在花括号中的任选的表达式。语法概要在13节中给出。 ​ 与本书第1版给出的语法不同,本书给出的语法使表达式运算符的优先级和结合性是显式的。     4 标识符的含义 标识符也叫名字,可以指代很多实体:函数、结构标记、联合和枚举,结构或联合的成员,枚举常量,类型定义名字以及对象。一个对象,有时也称为变量,是一个存储区域。它的解释依赖于两个主要属性:它的存储类和它的类型。存储类决定了与该标识的对象相关联的存储区域的生命周期,类型决定了该对象的值的含义。一个名字还有一个作用域和一个连接,作用域即程序中可见此名字的区域,连接决定另一区域中的同一个名字是否指代的是同一个对象或函数。作用域相连接将在11节今讨论。   4.1 存储类 存储类共有两类:自动存储类和静态存储类。一个对象说明的几个关键宇与上下文一起确定了对象的存储类。自动对象对于一个分程序(见9.3节)来说是局部的,在退出分程序时该对象将消失。如果没有提到存储类说明.或者如果使用了auto区分符,那么分程序中的说明生成的都是自动对象。说明为register的对象也是自动的,并且(在可能时)存储在机器的快速寄存器中。 静态对象可以局部于某个分程序,或者对所有分程序来说是外部的。不论是哪一种情况,在退出和再进入函数和分程序时其值不变。在一个分程序包括提供函数代码的分程序内,静态对象用关键字static说明。在所有分程序外部说明的且与函数定义在同一级的对象总是静态的,可以通过使用static关键字使它们局部于一个特定的翻译单元,这使得它们有内部连接。通过略去显式的存储类或通过使用关键字extern,可以使这些对象全局于整个程序,并有外部连接。   4.2 基本类型 有几种基本类型。在C标准库中描述的标准头文件定义了在局部实现中每种类型的最大值和最小值。C标准库给出的数表示最小的可接受限度。 说明为字符(char)的对象要大到足以存储执行字符集(execution character set)中的任何字符。如果字符集中的某个字符存储在一个char象中,那么该对象的值等于字符的整数码,并且是非负的。其他量也可存储在char变量中;但其取值范围,特别是其值是否有符号.依赖于具体的实现。 以unsigned char说明的无符号字符与普通字符占用同样的空间,但其值总是非负的。以signed char显式说明的有符号字符也与普通字符占用同样大的空间。 ​ 在本书的第1版中没有unsigned char类型,但它的用途很广泛。signed char是新增加的。 除了char类型外,还有3种不问大小的整数类型:short int、int和long int。普通int对象的大小与主机的自然结构一样大,其他大小的整数类型都有特殊的用途。较长的整数至少要占有与较短整数一样的存储空间;但是具体的实现可以便一般整数(int)有与短整数(short int)或长整数(long int)有同样的大小;除非特别说明,整数类型都表示有符号数。 以关键字unsigned说明的无符号整数遵守算术模2n的规则,其中n是相应整数表示的位数。这样对无符号数的算术运算永远不会溢出。可以存储在带符号对象中的非负值的集合是可以存储在相应的无符号对象中的值的子集,并且这两个集合的重叠部分的表示是一样的。 单精度浮点数float)、双精度浮点数(double)和多精度浮点数(long double)中任何类型可能是同义的,但精度由前到后是上升的。 ​ long double是新增加的类型,在第1版中long float与double类型等价,但现在已不再相同。 枚举是具有整型值的一个独特的类型。与每个枚举相关联的是一个有名常量的集合(见8,4节)。枚举类型类似于整数类型。但是,如果某个特定枚举类型的对象被赋于的值不是其常量中的一个,或者被赋于的不是一个同类型的表达式,那么枚举类型通常用于编译器以产生警告信息。 因为以上这些类型的对象可以被解释为数字,所以统称它们为算术类型。char类型、int族类型,不论大小如何,是否有符号.都统称为整数类型。类型float、double和long double统称为浮点类型。 void类型说明值的一个空集合,它被用来说明那些不产生任何值的函数的类型。   4.3 派生类型 除了基本类型外,我们还可以通过以下几种方法构造派生类型,这些派生类型从概念上说有无限多个:     给定类型的对象的数组;     近日给定类型的对象的函数;     指向给定类型的对象的指针;     包含一系列不同类型的对象的结构;     包含不同类型的几个对象中任意一个的联合。 ―般地,在构造对象时可以递归地使用这些方法。   4.4 类型限定符 对象的类型可以有附加的限定符。说明为const的对象表明此对象的值不可以改变。说明为volatile的对象表明它有与优化相关的特殊属性。限定符既不影响对象的值的范围也不影响其算术属性。限定符将在8.2节讨论。     5 对象和左值 对象是一个指名的存储区域,左值是指向某个对象的表达式。左值表达式的一个明显的例子是一个有合适类型与存储类的标识符。某些运算符可以产生左值。例如,如果E是一个指针类型的表达式,那么*E是一个左值表达式,它指代由E指向的对象。名字“左值”来源于赋值表达式E1 = E2,其中左运算分量E1必须是一个左值表达式。对每个运算符的讨论说明了此运算符是否需要一个左值运算分量以及它是否产生一个左值。     6 转换 依据运算分量的不同,某些运算符会引起运算分量的值由某个类型转换为另一个类型。本节解释这种转换所产生的结果。6.5节将讨论大多数普通运算符所需的转换;对每个运算符的讨论会在需要时做补充。   6.1 整提升 在一个表达式中,凡是可以使用整数的地方都可以使用有符号或无符号的字符、短整数和整数的位宇段及枚举类型的对象。如果原来类型的所有值都可用int类型表示,那么原来类型的值就被转换为int类型;否则就被转换为unsigned int类型。这一过程称为整提升。   6.2 整数转换 任何整数转换为某个给定的无符号类型的方法是:找出与此整数同余的最小的非负值,其模数为该无符号类型能够表示的最大值加l。在二进制补码表示中,如果该无符号类型的位模式较窄,那么这就相当于左截取:如果该无符号类型的位模式较宽,那么这就相当于对有符号值进行符号扩展和对无符号值填0。 当任何整数被转换成有符号类型时,如果它可以在新类型中表示出来则其值不变,否则它的值由具体实现定义。   6.3 整数和浮点数 当把浮点类型的值转换为整数类型时,其小数点部分将被丢弃掉。如果结果值不能用此整类型来表示,那么其行为是未定义的。特别地,将负的浮点数转换为无符号整类型的结果没有指定。 当把整类型的值转换为浮点类型时,如果该值在该浮点类型可表示的范围内但不能精确表示,那么结果可以是下一个较高的或下一个较低的可表示值。如果该值超过可表示的范围,那么其行为是未定义的。   6.4 浮点类型 当一个精度较低的浮点值被转换为有相同或更高精度的浮点类型时,它的值不变。当把一个有较高精度的浮点类型的值转换为精度较低的浮点类型时,如果它的值在可表示范围内,那么结果可以是下一个较高的或下一个较低的表示值。如果结果在范围之外,那么其行为是未定义的。   6.5 算术转换 许多运算符都会以相似的方式在运算过程中引起转换并产生结果类型。其效果是将所有运算分量转换为同一类型,并以此作为结果的类型。这种方式的转换称为普通算术转换。 1.​ 首先,如果一个运算分量为long double类型,那么另一个也被转换为long double类型。 2.​ 否则,如果一个运算分量为double类型,那么另一个也被转换为double类型。 3.​ 否则,如果一个运算分量为float类型,那么另一个也被转换为float类型。 4.​ 否则,同时对两个运算分量进行整提升,然后,如果一个运算分量为unsigned long int类型,那么另一个也被转换为unsigned long int类型。 5.​ 否则,如果一个运算分量为long int类型且另一个运算分量为unsigned int类型,那么结果依赖于long int类型是否可以表示所有的unsigned int类型的值。如果可以,那么unsigned int类型的运算分量转换为long int;如果不可以,那么两个运算分量均转换为unsigned long int类型。 6.​ 否则,如果一个运算分量为long int类型,那么另一个也被转换为long int类型。 7.​ 否则,如果―个运算分量为unsigned int类型,那么另一个也被转换为unsigned int类型。 8.​ 否则,两个运算分量均为int类型。 ​ 这里有两个变化。第一,对float运算分量的算术运算可以只用单精度而不是双精度;而在第1版中指定所有的浮点运算都是双精度。第二,当较短的无符号类型与较长的有符号类型一起运算时,不将无符号类型的属性传递给结果类型;而在第1版中无符号类型总是处于支配地位。新规则稍微有点复杂,但减少了当无符号数与有符号数混合使用时的麻烦。但当一个无符号表达式与一个同样大小的有符号表达式相比较时仍会得到不期望的结果。   6.6 指针和整数 指针值可以加上或减去一个整数类型的表达式,在这种情况下,整数表达式的转换按照对加法类运算符的讨论进行(见7.7节)。 两个指向同一数组中同一类型的对象的指针可以进行减法运算.其结果被转换为整数;转换方式按对减法类运算符的讨论进行(见7.7节)。 值为0的整常量表达式或强制转换为类型void *的表达式可通过强制转换、赋值或比较转换为另一种类型的指针。其结果将产生一个空指针,此空指针等于同一类型的另一空指针,但不等于任何指向函数或对象的指针。 某些其他涉及指针的转换也可进行,但其结果依赖于具体的实现。这些转换必须由一个显式的类型转换运算符或强制类型转换(见7.5节和8.8节)来指定。 指针可以转换为整数类型,只要此类型足够大;所要求的大小依赖于具体的实现。映射函数也依赖于实现。 一个整类型对象可以显式地转换为指针类型。映射总是使一个足够宽的从指针转换来的整数转换回到同一个指针,否则其结果依赖于实现。 指向某一类型的指针可以被转换为指向另一类型的指针,但是如果该指针不指向在存储区域中适当对齐的对象,那么结果指针可能会导致地址异常。指向某对象的指针在转换成一个指向其类型有更少或相同的存储对齐方式的限制的对象时,可以保证原封不动地再转换回来时。“对齐”的概念依赖于实现,但char类型的对象有最少的对齐限制。如将在6.8节中讨论的,指针也可以转换为void *类型,并可原封不动地转换回来。 一个指针可以转换为同样类型的另―个指针.除了增加或删除该指针所指的对象类型的限定符(见4.4节和8.2节)。如果增加了限定符,那么新指针与原指针等价,不同的是多了由限定符带来的限制。如果删除了限定符,那么对基本对象的运算仍受它实际说明中的限定符的限制。 最后,指向一个函数的指针可以转换为指向另一个函数类型的指针,调用转换后指针所指的函数的效果依赖于实现。但是,如果转换后的指针被重新转换为原来的类型,则结果与原来的指针一致。   6.7 空类型void 一个void对象的(不存在的)值不可以以任何方式使用,也不能被显式或隐式地转换为一非空类型。因为一个空表达式表示―个不存在的值,这样的表达式只可使用在不需要值的地方,例如作为一个表达式语句(见9.2节)或作为逗号运算符的左运算分量(见7.18节)。 可以通过强制类型转换将表达式转换为void类型。例如,在表达式语句中一个空的强制类型转换将丢掉函数调用的返回值。 ​ void没有在本书的第1版中出现,但是从本书第1版出版后,就一直被广泛使用着。   6.8 指向空类型void的指针 指向任何对象的指针可以被转换为void *类型而不会丢失信息。如果将结果再转换为初始指针类型,那么初始指针被恢复。与在6.6节中讨论的、一般需要显式的强制转换的指针到指针的转换不同,指针可以被赋值为void *类型指针.也可以赋值给void *类型指针.并和void *类型指针进行比较。 ​ 对void *指针的解释是新增加的,以前char *指针扮演通用指针的角色。ANSI标准特别允许void *类型指针和其他对象指针在赋值和关系表达中混用,而对其他的指针的混合使用则要求有显式的类型转换。     7 表达式 表达式运算符的优先级与本节中各小节的先后次序相同,即最高优先级的运算符最先介绍。例如,作为加法运算符+(见7.7节)的运算分量的表达式是在7.1节至7.6节定义的那些表达式。在每一小节中,各个运算符具有相同的优先级。在每个小节中也讨论了该节所讨论的运算符的左、右结合律。13节给出的语法结合了运算符的优先级和结合律。 运算符的优先级和结合律是明确 规定 关于下班后关闭电源的规定党章中关于入党时间的规定公务员考核规定下载规定办法文件下载宁波关于闷顶的规定 的,但是表达式的求值次序除少数例外情况外是没有定义的,尽管子表达式会有副作用。也就是说,除非一个运算符的定义保证了其运算分量以一特定顺序求值,否则具体的实现可以自由地选择任一求值次序,甚至可以交替求值。但是,每个运算符以与它所出现的表达式的句法分析兼容的方式将其运算分量产生的值结合起来。 ​ 这个规则取消了以前具有在数学上满足交换律和结合律的运算符的表达式可以任意排列的自由,但可能会在计算时不满足结合津。这个改变仅影响浮点数在接近其精确度限度的计算以及可能发生溢出的情况。 C语言没有定义在表达式求值过程中的溢出、除法检查和其他异常的处理。大多数现有C语言的实现在进行有符号整数表达式的求值时以及在赋值时忽略溢出异常,但并不是所有实现都这样做。对除数为0和所有浮点异常的处理,不同的实现有不同的方式,有时候可以用非标准库函数进行调整。   7.1 指针生成 对于某类型T,如果某表达式或子表达式的类型为“T的数组”。那么此表达式的值是指向数组中第一个对象的指针,并且此表达式的类型被转换为“指向T的指针”。如果此表达式是一元运算符&、++、--或sizeof的运算分量.或是赋值类运算符或圆点运算符.的左运算分量.那么转换不会发生。类似地,类型为“返回T的函数”的表达式被转换为类型“指向返回T的函数的指针”,除非此表达式被用作&运算符的运算分量。   7.2 初等表达式 初等表达式是标识符、常量、字符串或带括号的表达式。     初等表达式       标识符       常量       字符串       (表达式) 一个标识符只要是按下面所讨论的方式适当说明的就是初等表达式。其类型由说明指定。如果一个标识符指定一个对象(见5节)且其类型是算术、结构、联合或指针类型,那么它是一个左值。 一个常量是一个初等表达式,其类型依赖于它的形式,见2.5节的讨论. 一个字符串字面值是一个初等表达式。它的初始类型是char数组类型(对于宽字符字符串,则为wchar_t数组类型),但遵循7.1节给出的规则。它通常被修改为指向char类型(wchar_t类型)的指针,从而结果是指向字符串中第一个字符的指针。在一些初始化程序中不能进行这样的转换,详见8.7节。 用括号括起来的表达式是一个初等表达式,它的类型和值与无括号的表达式一致。此表达式是否是左值不受括号的影响。   7.3 后缀表达式 后缀表达式中的运算符遵循从左到右的结合规则。     后缀表达式:       初等表达式       后缀表达式 [ 表达式 ]       后缀表达式 ( 变元表达式表opt )       后缀表达式 . 标识符       后缀表达式 -> 标识符       后缀表达式 ++       后缀表达式 --     变元表达式表:       赋值表达式       变元表达式表, 赋值表达式 7.3.1 数组引用 带下标的数组由一个后缀表达式后跟一个括在方括号中的表达式来表示。这两个表达式中要有一个的类型必须为“指向T的指针”,其中T是某种类型;另一个表达式的类型必须为整数。下标表达式的类型为T。表达式E1[E2],在定义上等同于*((E1)+(E2))。有关数组引用的更多讨论见8.6.2节。 7.3.2 函数调用 函数调用由一个后缀表达式(称为函数命名符)后跟由圆括号括起来的包含一个可能为空的、由逗号分隔的赋值表达式表组成,这些表达式就是函数的变元。如果后缀表达式包含一个在当前作用域中不存在的标识符,那么此标识符就被隐式地说明,就好像说明     extern int 标识符(); 在包含此函数调用的最内层分程序中被给出一样。这个后缀表达式(在可能的隐式说明和指针生成之后,见7.1节)必须有类型“指向返回T的函数的指针”,其中T为某个类型,且函数调用的值的类型为T。 ​ 在第l版中,这个类型被限制为函数类型,并且在通过指向函数的指针来调用此函数时必须有一个显式的*运算符,ANSI C标准允许现存的一些编泽程序用同样的语法来进行函数调用和通过指向函数的指针来进行函数调用。旧的语法仍然可用。 术语变元用来表示传递给函数调用的表达式,而术语参数则用来表示由函数定义或函数说明所接收的输入对象(或其标识符),通常也可用术语“实际变元(参数)”和“形式变元(参数)”来区分它们。 在准备调用函数时,要对它的每个变元进行复制,所有的变元传递严格地按值进行。函数可能会改变其参数对象的值(即变元表达式值的拷贝),但这个改变不会影响变元的值。然而,可以将指针作为变元传递,以使函数可以改变指针所指向的对象的值。 函数可以用两种方式说明;在新的方式中,参数的类型是作为函数类型的一部分显式指定的,这种说明称为函数原型。在旧的方式中,参数类型没有说明。函数说明在8.6.3节和10.1节讨论。 如果在一个函数说明的作用域中函数是以旧方式说明的.那么按以下方式对每个变元进行缺省变元提升:对每个整类型变元进行整提升(见6.1节),将每个float类型的变元转换为double类型。如果调用时变元的数目与函数定义中参数的数目不等,或者某个变元的类型提升后与相应的参数类型不一致,那么函数调用的结果是未定义的。类型一致性依赖于函数定义是以新方式进行的还是以旧方式进行的。如果是旧方式的,那么类型一致性检查将在提升过的调用的变元类型和提升过的参数类型之间进行;如果定义是新方式的,那么提升过的变元类型必须与没有提升过的参数本身的类型一致。 如果在函数调用的作用域中函数说明是以新方式进行的,那么变元将被转换为函数原型中的相应参数类型,就像是赋值一样。变元数目必须与显式说明的参数数目相同,除非函数说明的参数表以省略号(, ...)结束。在这种情况下,变元的数目必须等于或超过参数的数目;其后无显式指定类型的参数与之对应的变元要进行缺省的变元提升,如前面段落中所述。如果函数定义是以旧方式进行的,那么在调用中可见的原型中的每个变元类型必须与相应函数定义中的参数类型一致(函数定义中的参数类型已进行过变元提升)。 ​ 这些规则特别复杂,因为必须要考虑到新旧方式函数的混合使用。应尽可能避免新旧方式混合使用。 变元的求值次序没有指定。不同的编译器的实现方式各不相同。然而,在进入函数前变元和函数命名符是完全求值的,包括所有的副作用。对任何函数都可以进行递归调用。 7.3.3 结构引用 一个后续表达式后跟一个圆点和一个标识符仍是一个后缀表达式。第一个运算分量表达式的类型必须是一个结构或联合,标识符必须是结构或联合的成员名字。结果值是结构或联合的指名的成员,其类型是对应成员的类型。如果第一个表达式是一个左值且第二个表达式的类型不是数组类型.那么整个表达式是一个左值。 一个后缀表达式后跟一个箭头(由-和>组成)和一个标识符仍是一个后缀表达式。第一个运算分量表达式必须是一个指向结构或联合的指针,标识符必须指名结构或联合的一个成员,结果指向指针表达式所指向的结构或联合的指名成员,结果类型是对应成员的类型。如果成员类型不是数组类型那么整个表达式是一个左值。 这样,表达式E1->MOS与(*E1).MOS等价。结构和联合将在8.3节讨论。 ​ 在本书的第1版中.已经规定了在这样的表达式中,成员的名字必须属于后缀表达式所指定的结构或联合,但是这个规则并没有强制实行。最新的编译程序和ANSI强制规定这一点。 7.3.4 后缀加一与减一运算符 一个后续表达式后跟一个++或--运算符仍是一个后缀表达式。表达式的值是运算分量的值。当执行完此表达式后,运算分量的值加l(++〕或减l(--)。这个运算分量必须是一个左值。对运算分量的限制和运算细节的详细讨论见加法类运算符(7.7节)和赋值类运算符(7.17节)。其结果不是左值。   7.4 一元运算符 表达式中的一元运算符遵循从右到左的结合原则。     一元表达式:       后缀表达式       ++ 一元表达式       -- 一元表达式       一元运算符 强制转换表达式       sizeof 一元表达式       sizeof(类型名)     一元运算符: 任意一个       & * + - ~ ! 7.4.1 前缀加一与减一运算符 以运算符++或--为前缀的一元表达式仍是一个一元表达式。运算分量将被加1(++)或减1(--),整个表达式的值是经过加减以后的值。该运算分量必须是一个左值。对运算分量的限制和运算细节的讨论详见加法类运算符(见7.7节)和赋值类运算符(见7.17节)。结果不是左值。 7.4.2 地址运算符 一元运算符&用于计算运算分星的地址。该运算分量必须是一个既不能指向位字段,也不能指向说明为register的对象的左值或函数类型。结果值是一个指针,指向由左值所指的对象或函数。如果运算分量的类型为T,那么结果的类型为指向T的指针。 7.4.3 间接寻址运算符 一元*运算符表示间接寻址,它返回其运算分量所指向的对象或函数。如果它的运算分量是一个指针且所指向的对象是算术、结构、联合或指针类型,那么它是一个左值。如果表达式的类型为“指向T的指针”,那么结果类型为T。 7.4.4 一元加运算符 一元+运算符的运算分量必须是算术类型,其结果是运算分量的值。如果运算分量是整类型,那么就要进行整提升.结果类型是经过提升后的运算分量的类型。 ​ 一元+运算符是ANSI C标准新增加的,增加它是为了与一元-运算符对称。 7.4.5 一元减运算符 一元-运算符的运算分量必须是算术类型,结果为运算分量的负值。如果运算分量是整类型,那么就要进行整提升。有符号数的负值的计算方式为:将提升所得到的类型中的最大值减去提升过的运算分量的值,然后加1;但0的负值仍为0。结果类型为提升过的运算分量的类型。 7.4.6 二进制求反运算符 一元~运算符的运算分量必须是整类型,结果为运算分量的二进制反码。在运算过程中要对运算分量进行整提升。如果运算分量是无符号类型的,那么结果是通过由提升后的类型的最大值减去运算分量的值得到的值。如果运算分量是有符号的,那么结果的计算方式为:将提升后的运算分量转换为相应的无符号类型,进行二进制求反运算,再将结果转换为有符号类型。结果的类型为提升后的运算分量的类型。 7.4.7 逻辑非运算符 运算符!的运算分星必须是算术类型或是一个指针。如果运算分量等于0,那么结果为1,否则结果为0。结果类型为int。 7.4.8 sizeof运算符 sizeof运算符用于求存储其运算分量类型的对象所需要的字节数。运算分量或者为一个未求值的表达式,或者为一个由括号括起的类型名字。当sizeof被用于char类型时,其值为1。当用于数组时,其值为数组中字节的总数。当用于结构或联合时,结果是对象中的字节数,包括任何使对象平铺为数组所需要的填充空间:有n个元素的数组的大小是一个元素大小的n倍。此运算符不能用于函数类型和不完全类型的运算分量,也不能用于位字段。结果是一个无符号整形常量,具体的类型由实现定义。在标准头文件(见C标准库)中,这一类型被定义为size_t类型。   7.5 强制转换 以括号括起来的类型名开头的一元表达式将导致表达式的值被转换为指名的类型。     强制转换表达式:       一元表达式       ( 类型名字 ) 强制转换表达式 这个结构称为强制转换。类型名字将在8.8节描述。转换的结果已在6节讨论过。包含强制转换的表达式不是左值。   7.6 乘法类运算符 乘法类运算符*、/和%遵循从左到右的结合规则。     乘法类表达式:       强制转换表达式       乘法类表达式 * 强制转换表达式       乘法类表达式 / 强制转换表达式       乘法类表达式 % 强制转换表达式 *和/的运算分量必须为算术类型,%的运算分量必须为整类型。对这些运算分量要进行常规算术转换,并预测结果类型。 二元运算符*表示乘法。 二元运算符/求得第一个运算分量被第二个运算分量除所得的商,而运算符%求得相应的余数。如果第二个运算分量为0,那么结果没有定义。其他情况下(a/b)*b + a%b等于a永远成立。如果两个运算分量均为非负,那么余数是非负的且小于除数,否则,仅可保证余数的绝对值小于除数的绝对值。   7.7 加法类运算符 加法类运算符+和-遵循从左到右的结合规则。如果运算分量有算术类型,那么要进行常规的算术转换。对于每个运算符有更多的可能类型。     加法类表达式:       乘法类表达式       加法类表达式 + 乘法类表达式       加法类表达式 - 乘法类表达式 运算符+作用的结果为两个运算分量的和。数组中指向一个对象的指针可以和一个任何整类型的值相加,后者将通过乘以所指对象的大小被转换为地址偏移量。相加的和是一个指针,它与初始指针有相同的类型,并指向同一数组中的另一个对象,此对象与初始对象之间有合适的偏移量。因此,如果P是一个指向数组中某个对象的指针,那么表达式P+1是指向数组中下一个对象的指针。如果相加的和所指定的指针不在数组的范围,且不是数组末尾的第一个位置,那么结果没有定义的。 ​ 允许指针指向救组的末尾是ANSI C新增加的,它使得我们可以像通常一样对数组元素建立循环。 运算符-作用的结果是两个运算分量的差值。可以从一个指针减去一个任意整类型的值,该运算的转换规则和条件与加法相同。 如果指向同一类型的两个指针相减,那么结果是一个有符号整类型数,表示所指向的对象之间的偏移量。相邻的对象之间的偏移量为1。结果的类型依赖于具体的实现,但在标准头文件中定义为ptrdiff_t。只有当指针指向的对象属于同一数组时,差值才有意义。然而,如果P指向数组的最后一个成员,那么(P+1)-P的值为1。   7.8 移位运算符 移位运算符<<和>>遵循从左到右的结合规则。每个运算符的运算分量必须为整类型,并且遵循整提升原则。结果的类型是提升过的左运算分量的类型。如果右运算分量为负值,或者大于等于左运算分量的位数,那么结果没有定义。     移位表达式:       加法类表达式       移位表达式 << 加法类表达式       移位表达式 >> 加法类表达式 E1<>E2的值E1右移E2个位。如果E1为无符号数或为非负值,那么右移等同于E1除以2E2。其他情况结果由具体实现定义。   7.9 关系运算符 关系运算符遵循从左到右的结合规则,但这个规则没有多大作用。a 移位表达式       关系表达式 <= 移位表达式       关系表达式 >= 移位表达式 当所指定的关系为假时,运算符<(小于)、>(大于)、<=(小于等于)和>=(大于等于)均返回0;当关系为真时,它们均返回1。结果的类型为int类型。如果运算分量为算术类型,那么要进行通常的类型转换。可以将指向同一类型的对象的指针进行比较(忽略任何限定符),其结果依赖于所指对象在地址空间中的相对位置。指针比较只对相同对象的部分有定义:如果两个指针指向同一个简单对象,那么它们的值相等;如果指针指向同一个结构的不同成员,那么指向后说明的成员的指针有较大的值;如果指针指向同一个联合的不同成员,那么它们有相同的值;如果指针指向一个数组的不同成员,那么它们的比较值等于对应的下标的比较值。如果指针P指向数组的最后一个成员,那么P+1比P大,尽管P+1已指向数组的界外。其他情况指针的比较没有定义。 ​ 这些规则允许指向同一个结构或联合的不同成员的指针进行比较,从而放宽了第1版所述的限制。这些规则还使得与指向正好超出数组末尾的指针的比较合法化。   7.10 相等类运算符     相等类运算符:       关系运算符       相等类运算符 == 关系运算符       相等类运算符 != 关系运算符 运算符==(等于)和!=(不等于)与关系运算符相似,但优先级不同。(任何时候只要a>= &= ^= |= 所有这些运算符要求左运算分量为左值,并且此左值可以修改,它不可以是数组、不完全类型或函数。同时其类型不能有const限定符,如果它是结构或联合,那么它的任意一个成员或递归子成员不能有const限定符。赋值表达式的类型是其左运算分量的类型,值是赋值发生后存储在左运算分量中的值。 在用=的简单赋值中,用表达式的值替换左值所指向的对象的值。下面几个条件中必须有一个条件成立:或者两个运算分量均为算术类型,在此情况下右运算分量的类型通过赋值转换为左运算分量的类型;或者两个运算分量为同―类型的结构或联合;或者一个运算分量是指针,另一个运算分量是指向void的指针;或者左运算分量是指针.右运算分量是值为0的常量表达式;或者两个运算分量均为指向同一类型的函数或对象的指针,除了右运算分量可能没有const或volatile说明。 形式为E1 op= E2的表达式等价于E1 = E1 op (E2),唯一的区别是前者E1只求值一次。   7.18 逗号运算符     表达式:       赋值类表达式       表达式 , 赋值类表达式 由逗号分开的一对表达式的求值次序为从左到右,并且左表达式的值被丢弃掉。结果的类型和值就是右运算分量的类型和值。在开始计算右运算分量以前,计算左运算分量所带来的副作用将被完成。在逗号有特殊含义的上下文中,如在函数变元表(见7.3.2节)和初始化符表(8.7节)中,所要求的语法单元是一个赋值表达式。这样逗号运算符仅出现在一个圆括号组中,例如,函数调用:     f(a, (t=3, t+2), c) 包含有三个变元,其中第二个变元的值为5。   7.19 常量表达式 从语法上看,常量表达式是局限于运算符的某一个子集的表达式:     常量表达式:       条件表达式 一些上下文要求表达式的值为常量,例如:在switch语句中的case后面、作为数组的边界和位字段的长度、作为枚举常量的值、用在初始化符中以及在某些预处理表达式中。 除非作为sizeof的运算分量,否则常量表达式中可以不包含赋值、增一或减一运算符、函数调用或逗号运算符。如果要求常量表达式为整类型,那么它的运算分量必须由整数、枚举、字符和浮点常量组成。强制类型转换必须指定为整类型,任何浮点常量都被强制转换为整数。此规则必须将数组、间接指向、取地址和结构成员运算排除在外(但是,sizeof可以带任何运算分量)。 初始化符中的常量表达式可以有更大的范围。运算分量可以是任意类型的常量,一元运算符&可以用于外部和静态对象以及以常量表达式为下标的外部和静态数组。一元运算符&也可以在出现无下标的数组或函数时被隐式地应用。初始化符计算的值必须或者为一个常量,或者为已说明的外部或静态对象的地址与一个常量的和或差。 允许在#if后面的整类型常量的范围较小,不可以是sizeof表达式、枚举常量和强制转换。参见12.5节。     8 说明 说明用于指定每个标识符的含义,它
本文档为【C语言标准参考手册】,请使用软件OFFICE或WPS软件打开。作品中的文字与图均可以修改和编辑, 图片更改请在作品中右键图片并更换,文字修改请直接点击文字进行修改,也可以新增和删除文档中的内容。
该文档来自用户分享,如有侵权行为请发邮件ishare@vip.sina.com联系网站客服,我们会及时删除。
[版权声明] 本站所有资料为用户分享产生,若发现您的权利被侵害,请联系客服邮件isharekefu@iask.cn,我们尽快处理。
本作品所展示的图片、画像、字体、音乐的版权可能需版权方额外授权,请谨慎使用。
网站提供的党政主题相关内容(国旗、国徽、党徽..)目的在于配合国家政策宣传,仅限个人学习分享使用,禁止用于任何广告和商用目的。
下载需要: 免费 已有0 人下载
最新资料
资料动态
专题动态
is_074439
暂无简介~
格式:doc
大小:363KB
软件:Word
页数:91
分类:互联网
上传时间:2011-10-31
浏览量:45