一、框架
1、Mac系统及常用工具、进制;C数据类型、常量变量、运算符、表达式、格式化输入输出
2、关系运算符、逻辑运算符、if、条件运算符、switch..case..
3、while循环、do..while循环、for循环
4、一维数组、数组排序、字符数组
5、二维数组、字符串数组、多维数组
6、函数分类、函数声明和定义、函数调用,形参实参、函数嵌套调用
7、结构体、结构体数组
8、指针
9、高级指针
10、程序存储区划分、动态申请内存、内存操作函数
11、函数指针、回调函数、动态调用
二、知识总结
一、基础:
1、进制、位权、1字节等于8位(位是计算机的最小储存单位,字节是计算机最小存储单元)、十进制转换其他进制:连除倒取余,其他进制转换十进制:按权求和
2、单行注释:// 多行注释: /*/ (内容写在量型号之间)
3、数据类型:整形类型、浮点类型、字符类型。
整型类型:长整形(long 4/8)、整形(int 4)、短整型(short 2)
浮点类型:单精度(float 4)、双精度(double 8)
字符类型:字符类型(char 1)
4、常量:程序运行期间不能被改变的量。
常量 类型:整型常量、浮点型常量、字符型常量
变量:实际上是储存区域,也可以做表达式,是程序运 行期间可以被改变的量。
变量定义:由类型符号、变量名、初始值三部分组成(如: int a = 5;)
变量命名规则:由数字、字母、下划线组成的,但不能 以数字开头,不能与关键字重复,不能重复使用变量名(同一作用域),见名知意。
5、赋值运算符:= 把右边的值赋给左边 优先级别最低
算数运算符:+, -, , / 整型两边相除取整,除数不能为0,% 两边均为正数 ,++ 自增运算符、- - 自减运算符:(它俩在前,先执行,它俩在后,后执行。
6、复合运算符:+= (a += b; 相当于a = a + b); -=; *=; /=; %=;(语法含义同上)
7、表达式:由常量、变量与运算符组合而成(注:表达式会有返回值)
8、输出函数:printf(); 输入函数 scanf()(注:以回车确定输入结束,只有大回车键才会结束;不要再格式串后面加’n’字符,否则会抵消人工输入的回车)。
转化符:%c->char; %d->int; %sd->short; %ld->long;%f->float、double。
9、转换格式控制:最少占4位:
printf(“%04d”,12); //不足补0
printf(“%4.2f”,1.2345); //保留小数点后两位
printf(“%-4d”,12); //左对齐
n回车符; 印 %%打印%;
二、运算符
1、BOOL类型:只有YES(1)和NO(0)两个值;
bool类型:由ture(非0数字)和flase(0)两种值;
BOOL和bool区别:1)、在C中只有小布尔没有大布尔; 2)、在C和OC中小布尔中非零为真,零为假;3)、在OC中大布尔中1为真,0为假,其它数字原样输出,不做判断条件
2、关系运算符:用于比较运算,结果值有真假两种,结果值由BOOL类型储存
> = <= == !=
3、逻辑运算符:由逻辑运算符组成,结果也是非真即假
&&(逻辑与):一假即假(有一边为假则结果为假)
||(逻辑或) :一真即真(有一边为真则结果为真)
!(逻辑非) :对表达式结果取反
4、条件运算符:条件表达式?表达式1:表达式2
如果条件为真,则执行表达式1。 如果条件为假,则执行表达式2。
5、枚举类型:属于构造类型,是一组由符号的整型常量,一一罗列所有状态,罗列所有结果。 enum枚举名{枚举值表}
目的和作用:提高程序可读性;将人能看懂的标示符与计算机能识别的数字建立对应关系
三、程序的三种结构:
1、顺序结构:按照语句顺序执行
2、分支结构:判断条件在两个语句序列中选择,一旦进行判断必须选择其中一个。
if语句:形式一:if (条件表达式) {语句;}
形式二:if (条件表达式) {语句1;} else{语句2;}
形式三:if (条件表达式1) {语句1;} else if条件表达式2){语句2;} else{语句3;}
switch语句:多分支语句,通过判断整形表达式的值,来判断执行那个语句。常与case配合使用。
break:跳出本层循环,如果外层还有循环的话,则执行外层循环,如果外层没有循环,则执行循环后面的代码。(常与if、switch连用)
continue:跳出本次循环,跳出本次循环,遇到continue后,不管continue后面有无代码,均不执行,提前结束当前循环,开启下一次循环。(常与if连用)
arc4random():返回随机数的函数。 如果要一个随机数在【a, b】范围内的整数。公式为:arc4random() % (b - a +1) + a;
3、循环语句:在满足一个条件的情况下反复执行一个语 句序列。
while语句:while(条件表达式){语句;}条件表达式为真执行循环体,一旦为假立即停止执行。
do…while…语句:do{语句;} while (条件表达式),先执行循环体,再判断循环条件直到不满足,才结束循环。
for语句:for(循环变量初始化;循环条件;循环增量){语句;}循环条件为真才执行循环体。
四、一维数组
一维数组:属于构造结构,是由相同类型的成员组成的一组数据。
数组名:存储不同数据类型的数组有各种对应的名称(名称可以随意起),但 要见名知意。
元素:数组由多个成员组合而成,每个成员被称为“数组的元素”
1、定义一个数组
int array[3] = {1, 2, 3};
int:声明数组类型 array:数组名 [5]:数组长度(注:定义数组时,可以省略元素个数,因为编译器会自动推断出元素个数,但是“[]”坚决不能省略。) { } :给数组赋值,元素之间用逗号分隔开(注:1)定义数组时,可以不给每一个元素赋值,没有赋值的元素系统会默认为0。2)定义数组时,给定了数组长度,首元素为0,其他元素不赋值,则默认为0,这样五个元素均为0。3)大括号中最好赋上初始值,因为没有赋值是系统会根据Xcode编译环境来赋值,有时可能有,有时可能无,这样就会出现定义不明,所以建议赋上值。)
2、数组的内存占用
1)使用sizeof()函数求数组占字节数
printf("array[3]数组所占字节数%ld", sizeof(array));
结果:12 因为每个int类型占4个字节,有3个元素,所以共12个字节 。
2)利用sizeof()函数求数组元素个数
先求出数组所占字节数,再求出其中某个元素所占字节数,再相除就得出数组元素个数。
int array[5] = {4, 6, 7, 2, 8};
int a = (sizeof(array) / sizeof(array[1]));
printf("a = %dn", a); 结果为:a = 5
3、数组元素的访问方式:数组名+下标
1)int array[6] = {12, 21, 32, 23, 25};
访问array中的32这个元素: 数组名:array,下标:从零开始计数,32在第3个位置上,所以32的下标是2。
int temp = array[2];
printf("temp = %d, array[2] = %dn", temp, array[2]); 结果:32,32
(注:可以直接访问,打印出来,也可以把值赋给另一个相同类型的变量再打印出来)
2)数组下标越界:数组下标超过了数组长度,访问了不属于数组的内存空间。在C中是没有“数组下标越界保护”机制,所以在写代码时,要避免下标越界。
3)修改数组元素
3.1)修改array中的第二个元素为62
int array[6] = {12, 21, 32, 23, 25};
array[1] = 62;
printf("array[1] = %dn", array[1]); 结果:array[1] = 62
3.2) 定义一个新的数组,array1,将array的元素,原样赋给arrary1.
int array[5] = {12, 21, 32, 23};
int array1[5] = {0};
for (int i = 0; i < 5; i++) {
array1[i] = array[i]; // 数组的下标可以是常量,也可以是变量
printf("array1[%d] = %dn", i, array1[i]); // 数组的输出需要for循环进行遍历
}
// array1 = array; 数组中不能整体访问数组元素,也不能整体赋值,而要逐一的遍历数组元素,进行访问、赋值、输出。
// 定义一个具有20个元素的数组,每个元素的取值范围是【30,70】,求数组元素的和
// 1.定义一个整型数组长度为20
int array[20] = {0};
//2.开启一个for循环,使用循环给数组元素赋值
for (int i = 0; i < 20; i++) {
//2.1 生成【30,70】之间的数
int temp = arc4random() % (70 - 30 + 1) + 30;
//2.2 将生成的随机数赋值给对应的元素
array[i] = temp;
printf("array[%d] = %d\n", i, array[i]);
}
//3.求数组元素的和
//3.1定义一个保存和的变量
int sum = 0;
//3.2 使用for循环遍历数组元素,同时将元素累加起来
for (int i = 0; i < 20; i++) {
// 3.2.1 获取当前数组元素并将获取到的元素加起来
sum = sum + array[i];
}
printf("元素总和为%d\n", sum);
// 定义三个数组,将随机产生的数放入其中两个数组中,将这两个数组的值加起来放入第三个数组中
// 1、 定义三个数组,并初始化。
int array1[10] = {0};
int array2[10] = {0};
int array3[10] = {0};
// 2、 使用一个for循环,给第一个数组和第二个数组赋值,将两个数组个元素加起来,交给第三个元素。
for (int i = 0; i < 10; i++) {
// 2.1、 分别把生成的随机数赋给两个数组
array1[i] = arc4random() % (40 - 20 + 1) + 20;
array2[i] = arc4random() % (40 - 20 + 1) + 20;
// 2.2、两数组进行相加
array3[i] = array1[i] + array2[i];
// 3、 打印验证结果是否正确
printf("array1[%d] = %d, array2[%d] = %d, array3[%d] = %d\n", i, array1[i], i,array2[i], i,array3[i]);
}
4、冒泡排序
int array[5] = {12, 45, 23, 42, 56};
//趟数 = 元素个数 - 1;每趟比较的次数 = 元素个数 - 当前的趟数
//两层for循环,外层丛植比较趟数、内层控制比较次数
//趟数
for (int j = 0; j < 5 - 1; j++) {
//次数
for (int k = 0; karray[k + 1]){//因为每趟比较都要从下标为0开始,j在每次从外层循环进入内层时,都要被初始化为0。因为每次比较,下标都要加一,j在每次内层循环时,也会加一,所以,在这里是用j做下标是没有问题的。
//满足交换条件,开始交换位置。
//1.定义一个临时变量
int temp = array[k];6
array[k] = array[k + 1];
array[k + 1] = temp;
}
}
}
for (int i = 0; i < 5; i++) {
printf("%d ", array[i]);
}
5、字符数组
字符数组:保存字符的数组就是字符数组。有’0’结束标识的字符数组也称为字符串。
定义一个字符数组:声明类型、数组名、数组长度、赋初值
char char_Array[5] = {'h', 'e', 'l', 'l', ‘o’};
//每个字符占一个字节,数组长度是5,所以数组占5*1个字节的内存。
char char_Array[5] = {'h', 'e', 'l', 'l', 'o', '0'};和char char_Array[5] =“hello”;
是同一个意思都便是定义字符串hello。
字符串和字符数组的区别:
如果字符数组最后一个字符末尾没有‘0’,则是一个字符数组
如果字符数组最后一个字符末尾有‘0’,则是这个字符数组就是字符串。
字符串操作函数
strlen():返回字符串的有效字符个数
strcpy():字符串拷贝函数:把第二个参数的值拷贝给第一个参数
strcat() 字符串拼接:把第二个参数的值,拼接到第一个参数的后面
strcmp() 字符串比较:第一个参数大于第二个参数返回一个正整数,第一个参数小于第二个参数,返回一个负整数,两个参数相等,返回零。
五、函数
1、函数:完成某一功能的代码段。
函数名:给这段代码取一个名字,以后就可以通过这个名字来代替这段代码,遵守变量名的命名法则。
函数参数: 有的函数在完成某一任务时需要的材料,我们把在完成任务所需的材料叫函数的参数。 在被调函数中,参数叫“形参”,在主调函数中,参数叫“实参”。
函数参数传递:将实参的值拷贝一份到形参变量的空间中,在被调用函数中操作的参数是新开辟的内存的数值,和实参没有关系。
参数列表:如果没有参数,可以用void修饰,也可以什么都不写,但是小括号不能省略。
函数的返回值:有的函数在完成任务之后,需要把任务结果返回给主调函数,就可以通过return语句来将值返回。注意:return只能返回一个值,同时不能返回数组。
return:return标志着函数的结束,如果return后边有数值,则将这个数值交给主调函数。当return只负责结束函数执行时,可以省略return。return后⾯的语句不再执行。
函数定义:一般定义在main函数前,函数会遭遇嵌套定义。
返回值类型 函数名 (参数列表) {函数实现……return 返回值;}
函数的调用:必须要先定义出来这个函数,定义之后才能调用
函数调用方式:函数名+参数列表
函数类型:无返无参,无返有参,有返无参,有返有参
返回值类型:如果无返回值,必须要使用void修饰符
2、.h和.m文件
1)如果使用多个文件来管理定义函数时,函数的声明要放在.h文件中,函数定义在.m文件中。
2)由于函数的声明和定义在其他文件中,所以想使用该函数,必须先引入声明部分,也就是.h文件
import"文件名.h" 把这个放在 主函数和系统头文件之间
3) 如果函数定义在主调函数(main)函数之前,函数的声明可以省略,因为在编译时,编译到主调函数时,已经知道该函数被调函数的意义,所以不会出现异常。如果函数定义在主调函数(main)函数之后,务必要在main函数之前对函数进行声明。
4)声明的意义在于:告诉系统,后面将会出现一个该类型的函数。一个完整的自定义函数,应该包括:函数声明、函数定义、函数调用,共三部分。
5)函数声明的方法:将函数的“返回值类型+函数名+形参列表+;”写在导入头文件语句下面,就是一个函数声明。
3、函数的嵌套调用
1)函数可以嵌套调用:在一个函数中,调用另一个函数完成某项功能,这就是函数的嵌套调用。函数可以嵌套调用,但是函数不可以嵌套定义。
2)函数不可以嵌套定义:在定义一个函数时,又定义另一个函数,就叫函数的嵌套定义,这种行为是不允许的。
4、函数的递归调用
函数递归时一定要有出口,否则会造成程序递归环路(死递归),那么程序一定会崩溃。递归调用:自己调用自己完成某项功能。
5、变量的作用域
// 局部变量:只在当前作用域中起效
// 全局变量:不属于任何一个作用域的变量就是全局变量,全局变量可以在当前文件中任何一个地方被访问到(从创建的位置开始,一直到文件结束)
// 如何确定作用域?使用{}来确定作用域,一对大括号,就确定了一个作用域。
六、结构体
1、什么是结构体?
1、)结构体是一种构造类型,由其他基本数据类型组合而成一种新类型。与数组的区别是:构成结构体的基本数据类型可以不相同,数组必须是相同类型的数据构成。
2、)结构体必须先声明,声明结构体由哪些数据类型构成的,相当于基本数据类型的“int/char/double”之类的类型说明。
3、)声明完成后,就在文件中创造出了这种新的类型,就可以使用这种新的类型定义变量来保存数据了。
2、结构体类型的声明
struct+结构体名+{成员标量}+; 结构体声明在.h文件或主函数与系统头文件之间。
eg:struct student {
char name[20];
int age;
char gender;
};
3、结构体定义
1)struct关键字+结构体名,共同组成了变量的类型修饰符。
struct+结构体名+结构体变量名 = {值1,值2};
eg:struct student stu1 = {"zhangSan", 23, 'm'};
2)结构体成员变量的访问
方式:使用结构体变量名+“.”+”成员变量”的方式访问
eg:访问姓名:stu1.name 访问年龄:stu1.age 访问性别:stu1.gender
3)结构体成员变量的修改
先访问到对应的成员变量,然后再修改值
修改成员年龄:stu1.age = 22;
修改成员性别:stu1.gender = ‘f’;(输出打印时用%c)
修改成员姓名:stu1.name = "shuKui"; (这是错误的形式)
因为字符串在C中使用时,字符数组类型保存的数组不能整体赋值,所以使用普通的修改成员变量方法在这里无效,需要使用字符串拷贝函数,达到修改的目的。
strcpy(stu1.name, “shuKui"); (使用strcpy()函数进行字符串的复制替换)
4、结构体数组
保存结构体变量的数组,就是结构体数组。结构体数组和基本数据类型的数组一样,只能保存相同的数据元素(结构体变量类型必须相同)
1)声明一个father类型的结构体数组
typedef struct father {
char name[20];
int age;
}father;
2)定义一个father类型的结构体数组
father father[4] = {
{"zhangsan", 12},
{"laowang", 55},
{"xiaotian", 66},
{"xiaoyan", 22}
};
3)访问成员变量的方式:数组变量名 + 下标 + ‘.’ + 成员变量名
访问姓名:father[0].name
访问年龄:father[2].age
七、指针
1、指针:指针就是内存单元编号的另一种称呼。每个内存单元都有一个编号,这个编号就是内存单元的地址,也被称作指针。
指针变量:用来保存“内存单元”地址的变量,就是指针变量。
2、定义一个指针变量
1)int p = NULL; int a = 12;
// int : 类型声明符,表示变量p是一个整型指针变量。
// int:告诉系统取值时,应该从当前位置取多少字节数据。int表示连续取4个字节。它与指针变量所占字节无关,指针变量所占字节数只与操作系统有关,64位环境下占8字节,32位环境下占4字节。
// 不能使用指针计算数组元素的个数,因为指针所占的内存的字节数至于操作系统的位数有关
// p:变量名,一般我们将指针类型的变量取名叫p。
// NULL:C语言中,使用NULL代替“空”的意思,将NULL赋给指针,则指针此时保存的是内存起始地址,指针指向内存的起始位置。
printf("%pn", p); // 打印结果0x0,表示内存的起始位置。
// 使用%p作为占位符,来打印指针变量所保存的地址。
2) &:取地址符号,将后边变量的地址取出来,交给指针保存。
将整型变量a的地址,交给整型指针变量p保存。
p = &a; printf("%pn", p);
3):取值运算符,将指针所指的内存地址中保存的数据取出来。
注意:在不同的位置的含义不同,出现在指针定义的位置,表示证明后边的变量名是一个指针。如果前边没有类型修饰符,则表示指针所指向的地址的数值。
int b = p; printf("%dn", b);
4)用指针修改变量 a 的值
*p = 23; printf("%dn", a);
5)指针重指向:就是指针把它里面保存的地址进行个更换。
p = &c; 指针发生了重指向,有原来的指向a改变为指向b。
3、指针的算术运算
p++; // 指针变量进行加法操作,指针所保存的地址将会加若干个单位的字节,每个单位占用的字节数取决于指针的类型。指针从低地址向高地址移动。
p- -; // 指针变量进行减法操作,指针所保存的地址将会减若干个单位的字节,每个单位占用的字节数取决于指针的类型。指针从高地址向低地址移动。
// 普通的加减运算:如果没有小括号干预优先级,则先取值,再将值进行加减操作
// 自增、自减运算:先处理自增、增减,处理完成后,再进行取值操作。
4、指针与数组
1)数组名就是数组的首元素地址,我们可以定义一个指针变量,来存储数组的首地址,这样指针变量指向的就是这个数组。
// 定义一个指针,将指针指向数组array的首地址
int p = array; // 数组名本身就是地址,所以不需要取地址符&
// 访问数组元素:数组名+下标针+下标
printf("array[1] = %dn", array[1]);
printf("p[1] = %dn", p[1]);
// 打印指针指向的地址和数组首元素的地址以及数组名指向的地址
printf("%pn", p);
printf("%pn", &array[0]);
printf("%pn", array);
2)数组名和指针的区别:
2.1)指针指向数组的首地址后,可以向数组名一样访问数组元素(数组名/指针);
2.2) 指针变量的地址可以被修改,但是数组名是一个常量,它所保存的地址不能被修改(数组名不能重指向,但是指针可以重指向)
3)// 使用指针的方式来访问数组元素
int temp = p; int temp1 = (p + 1); int temp2 = (array + 1);
// 此时指针P的指向并没有发生改变,而是产生了一个"新指针(p + 1)"指向了第二个元素。
// 通过修改指针P的指向,来访问数组元素;
int temp3 = (++p); printf("temp3 = %dn", temp3);
int temp4 = (++array); // 因为数组名是常量,所以不能通过修改数组名的地址修改数组元素。(这种表示是错误的)
5、使用指针遍历数组
int array[5] = {23, 12, 34, 23, 45};
int *p = array;
6、指针与字符串
1)// 定义一个字符数组用来保存字符串"hello"
char string[] = “hello"; // hello 是存储在常量区的,string将hello拷贝一份放到栈区进行使用。
// 定义一个char类型指针指向string
char *p = string; //这种情况是指针指向有string开辟的栈区的hello的首地址
// 使用指针下标为0的元素改为’a’
p = ‘a’;
// 使用指针操作字符串:输出字符串,使用%s做占位符,参数用指针名,或数组名均可。
printf("%sn", p);
// 使用指针便利字符串
for (int i = 0; i < strlen(string); i++) {
printf("%cn", (p+i));
}
2)// 指针p此时指向常量区,所以不能修改常量字符串的值。常量字符串,只能访问不能修改。
// char p1 = “hfjhsdjfh"; // 这种情况是指针指向常量区
// printf("%sn", p1);
// 总结:指针指向栈区的字符串,可以访问进行修改,但是当指针指向常量区时,只能访问不能修改。
7、指针数组
1)char str[3] = {"ios", "ipad", “imac”};这是指针数组
// str[0],相等于第一个指针,指向ios首地址。
// str[1],相当于第二个字符指针,指向常量区的ipad字符串的首地址。
// str[2],相当于第三个字符指针,指向常量区的imac字符串的首地址。
char srt3 = {"ios", "ipad", “imac"};这是二位数组(两个表达意思一致)
// char 类型的数组里存的并不是三个字符串,而是三个字符串的首地址。如果用数组存储三个字符串,需要使用二维数组。
// 指针数组中保存的元素都是指针。
// str本质是一位数组,str数组中存储的时char类型的指针。
2)访问
printf("%sn", str[1]); // 这就访问了str[3]中的第二个元素(字符串)
3)修改:修改指针数组的元素。在这里,就是修改指向。
str1[2] = "Happy Birthday";
printf("%sn", str1[2]);
4)使用指针打印:1、字符串的地址2、一个完整的字符串3、字符串中,下标为3的字符
// char p2 = "i love";
// printf("%pn", p2);
// printf("%sn", p2);
// printf("%cn", (p2+3));
// 打印ipad中的第二个字符。
char str2[3] = {"ios", "ipad", “imac"}; printf("%cn", (str2[1]+1));
8、指针作为函数的参数
// 对两个整型数值进行交换
// 1、函数exchangeNum定义
void exchangeNum(inta, intb){ 传入的参数必须是指针(地址),这里坚决不能传入数组名,虽然数组名是地址,但是数组是常量怒能被改变所以不能传。(用指针传参本质是把要交换的变量的地址传过去)
int temp = *a;
a =b;
*b = temp;
}
int main(int argc, const char * argv[]) {
int x = 23, y = 88;
int *p1 = &x;
int *p2= &y;
// 调用exchangeNum函数
exchangeNum(p1, p2);
printf("x= %d, y = %dn", x, y);
return 0;
}
9、结构体指针
指向结构体变量的指针就是结构体指针。结构体指针的类型必须和结构体变量的类型一致。
// 声明一个Student类型的结构体变量
typedef struct student {
char name[20];
int age;
float score;
char gender;
}Stu;
// 定义一个Student类型的结构体变量,并赋值
Stu stu1 = {"xiaoZeMaLiYa", 22, 98.15, ‘f'};
// 使用结构体变量名+.+成员变量名的方式访问
printf("%sn", stu1.name);
printf("%cn", stu1.gender);
// 定义一个指针,来指向这个结构体变量
Stu *p = &stu1; // 结构体变量名只是一个名称所以需要取地址符
// 使用指针访问结构体变量的成员变量
// 方法一:先取出结构体变量的值,再通过“.”找到它的成员变量
printf("%sn", (p).name); // 由于.运算的优先级高于,所以需要用小括号干预他们的运算顺序。
printf("%dn", (*p).age);
// 方法二:使用指针名+ -> 访问结构体的成员变量。
printf ("%sn", p->name);
printf("%dn", p->age);
// 访问name成员变量的首个字符及第六个字符
printf("%cn", *(p->name));
printf("%cn", *(p->name+5));
10、指针与结构体
结构体数组的数组名就是一个结构体指针,只不过数组名是一个常量,不能修改数组名保存的地址。
typedef struct teacher{
int num;
char name[20];
char gender;
float score;
}Teacher;
Teacher *stuu = student;
// 使用数组名+下标的方式,访问第一个元素的name成员变量
printf ("%sn", student[0].name);
// 使用指针取值的方式来访问第二个元素的num成员变量(不常用)
printf("%dn", (*(stuu+1)).num);
// 使用指针和 -> 的方式访问第三和元素的gender成员变量
printf("%cn", (stuu+3)->gender);
八、宏
1、普通宏
define kMac 33 // 宏定义:在程序,只要遇到kMac,在编译之前会自动替换为后面的数。 宏不是变量
命名规则: 纯大写或 k + 驼峰法
2、带参宏
define SUM(A,B) ((A) * (B)) // 带参宏只是将参数原样替换所以一般情况下,要在表示参数的字符上,添加小括号,防止出现运算符优先级别的问题。
// 带参宏和函数区别: 带参宏会增加预编译的时间,但是不会加大CPU的运算速度;函数虽然在预编译阶段非常快但是占用的CPU内存会加大,增大硬件损耗。
3、条件编译指令
作用:按条件不同,编译不同的代码
1)第一种形式:如果宏定义了标示符,则编译第一段代码,否则编译第二段代码
ifdef 标识符
代码段1
else
代码段2
endif
2)第二种形式:如果没有定义表示符,则编译代码段1,否则编译代码段2
ifndef 标识符
代码段1
else
代码段2
endif
3)第三种形式:如果 常量表达式结果非0 编译器编译代码1,否则编译代码段2
if 常量表达式
代码段1
else
代码段2
endif