黑马程序员之ios学习总结——07 C语言的枚举、预处理指令

今天下午复习完结构体类型,晚上复习下C语言中的另一种数据类型---枚举类型。枚举类型在iOS中也是很常用的。

一:枚举

枚举是C语言中的一种基本数据类型,并不是构造类型,它可以用于声明一组常数。当一个变量有几个固定的可能取值时,可以将这个变量定义为枚举类型。比如,你可以用一个枚举类型的变量来表示季节,因为季节只有4种可能的取值:春天、夏天、秋天、冬天;或者是定义一个枚举类型来表示一个人的性别属性,是男或者是女。

枚举类型的定义

一般形式为:enum 枚举名 {枚举元素1,枚举元素2,……};

enum Season {spring, summer, autumn, winter}; 

枚举变量的定义:

1、跟结构体一样,有3种方式定义枚举变量
     enum Season {spring, summer, autumn, winter};
enum Season s;


2.定义枚举类型的同时定义枚举变量
     enum Season {spring, summer, autumn, winter} s;


3.省略枚举名称,直接定义枚举变量
     enum {spring, summer, autumn, winter} s;

枚举使用的注意:

1 C语言编译器会将枚举元素(spring、summer等)作为整型常量处理,称为枚举常量。
2 枚举元素的值取决于定义时各枚举元素排列的先后顺序。默认情况下,第一个枚举元素的值为0,第二个为1,依次顺序加1。

enum Season {spring, summer, autumn, winter};
//也就是说spring的值为0,summer的值为1,autumn的值为2,winter的值为3

二:预处理指令

1.C语言在对源程序进行编译之前,会先对一些特殊的预处理指令作解释(比如之前使用的#include、#import文件包含指令),产生一个新的源程序(这个过程称为编译预处理),之后再进行通常的编译
2.为了区分预处理指令和一般的C语句,所有预处理指令都以符号"#"开头,并且结尾不用分号
3.预处理指令可以出现在程序的任何位置,它的作用范围是从它出现的位置到文件尾。习惯上我们尽可能将预处理指令写在源程序开头,这种情况下,它的作用范围就是整个源程序文件
4.C语言提供的预处理指令主要有:宏定义、文件包含、条件编译

不带参数的宏定义
  1.一般形式
#define 宏名 字符串
比如#define ABC 10
右边的字符串也可以省略,比如#define ABC

2.作用
它的作用是在编译预处理时,将源程序中所有"宏名"替换成右边的"字符串",常用来定义常量。

3.使用习惯与注意
① 宏名一般用大写字母,以便与变量名区别开来,或者前面用小写字母k开头。
② 对程序中用双引号扩起来的字符串内的字符,不进行宏的替换操作。
③在编译预处理用字符串替换宏名时,不作语法检查,只是简单的字符串替换。只有在编译的时候才对已经展开宏名的源程序进行语法检查

④ 宏名的有效范围是从定义位置到文件结束。如果需要终止宏定义的作用域,可以用#undef命令

带参数的宏定义:

1.一般形式
#define 宏名(参数列表) 字符串

2.作用
在编译预处理时,将源程序中所有宏名替换成字符串,并且将 字符串中的参数 用 宏名右边参数列表 中的参数替换

 #include 
 
  #define average(a, b) (a+b)/2 // 定义了一个带有2个参数的宏average
  
  int main ()
  {
      int a = average(10, 4); //会被替换成:int a = (10 + 4)/2
     
      printf("平均值:%d", a);
    return 0;
 }

3.使用注意

宏名和参数列表之间不能有空格,否则空格后面的所有字符串都作为替换的字符串
带参数的宏在展开时,只作简单的字符和参数的替换,不进行任何计算操作。所以在定义宏时,一般用一个小括号括住字符串的参数。

4.与函数的区别
从整个使用过程可以发现,带参数的宏定义,在源程序中出现的形式与函数很像。但是两者是有本质区别的:
① 宏定义不涉及存储空间的分配、参数类型匹配、参数传递、返回值问题
② 函数调用在程序运行时执行,而宏替换只在编译预处理阶段进行。所以带参数的宏比函数具有更高的执行效率


预处理指令2-条件编译

在很多情况下,我们希望程序的其中一部分代码只有在满足一定条件时才进行编译,否则不参与编译(只有参与编译的代码最终才能被执行),这就是条件编译

。 #if 条件1
  ...code1...
#elif 条件2
  ...code2...
 #else
  ...code3...
 
#endif

① 如果条件1成立,那么编译器就会把#if 与 #elif之间的code1代码编译进去。
② 如果条件1不成立、条件2成立,那么编译器就会把#elif 与 #else之间的code2代码编译进去
③ 如果条件1、2都不成立,那么编译器就会把#else 与 #endif之间的code3编译进去
④ 注意,条件编译结束后,要在最后面加一个#endif,不然后果很严重(自己思考一下后果)
⑤ #if 和 #elif后面的条件一般是判断宏定义而不是判断变量,因为条件编译是在编译之前做的判断,宏定义也是编译之前定义的,而变量是在运行时才产生的、才有使用的意义

#if defined()和#if !defined()的用法:
#if 和 #elif后面的条件不仅仅可以用来判断宏的值,还可以判断是否定义过某个宏。

#if defined(MAX)
     ...code...
#endif

如果前面已经定义过MAX这个宏,就将code编译进去。它不会管MAX的值是多少,只要定义过MAX,条件就成立。
条件也可以取反:
#if !defined(MAX)
      ...code...
#endif

#ifdef和#ifndef的使用:
#ifdef的使用和#if defined()的用法基本一致
#ifdef MAX
     ...code...
  #endif

#ifndef又和#if !defined()的用法基本一致
#ifndef MAX
     ...code...
#endif

总结:条件编译看上去和选择结构的if语句相似,但是两者是有本质区别的。条件编译#if是编译进去,不是执行,很平时用的if-else是不一样的)




你可能感兴趣的:(黑马程序员之ios学习总结——07 C语言的枚举、预处理指令)