C: 基本语法
C
预处理
介绍
在编译一个.c文件时,最开始的操作便是预处理,它会将包含的文件展开,把使用过宏的地方替换为宏定义的语句,删除注释等等…
可以用gcc生成一个预处理后的文件(后缀为.i)
1 | -E选项表示在预处理后停止,-C表示保留库函数里的注释,-o表示自定义输出文件的名称 |
宏
宏,如其名,它将一串命令序列转换为简短的语句,用短短几个单词即可以实现强大复杂的功能
预处理器对宏实际的操作是把参数作为字符串来替换并把这条替换后的语句插入程序控制流里
运算符
\(宏延续)
它用于隔开两个语句,当宏太长时,可用'\'分隔为多行
#(字符串常量化)
它可以将宏参数转换为字符串,字符串的内容是宏参数的名称,如要在字符串中转换需要再加上一层双引号
##(标记粘贴)
它可以将两边的标记合并成一个标记
defined(已定义)
它接受一个标识符,若这个标识符已定义,返回true,否则返回false
处理目标
#include
各种语言都有官方的函数库,这些库包含了许多优化度极高或十分方便的宏,想要使用它,我们需要声明
1 |
上述语句包含了一个名为stdio.h的文件,如果使用gcc生成一个预处理后的文件(后缀为.i),我们会发现这一文件条目很多,正是展开这一文件的效果
我们也可以用双引号包含,这时预处理器会先从当前目录而不是从库函数目录里寻找
1 |
在我们可以编写自己的头文件时,需要用这种包含方式包含
#define
这是定义一个常量,或是一个简易函数等的方法
1 |
上述语句的第一条定义了一个名为COL,大小为50的常量
第二条语句定义了一个PRINT宏函数,传入(int)n,打印"(n的名称)是(n的数值)"
双引号括住#+宏参数名可以解析它为它的名称
第三条语句的宏参数为"...",它可以接受多个参数,并把这些参数传给__VA_ARGS__
第四条语句被用于批量创建变量,##即黏合剂,将两边记号黏着
1 | int GIVE_NEW(3)= 5; // 创建一个名为"aNum3",值为5的int型变量 |
需要注意的是,当函数含参时,需要加上()括住,原因是传参时它将表达式参数当作字符串处理,加上括号才可以达到想要的结果
1 |
|
使用场景
宏是内联函数,即在预处理阶段时会把相应语句直接插入到程序中
与一般函数不同,它不需要跳转,跳出,而只需要牺牲一定的空间;而一般函数会被保存为副本,每次调用都必须跳到入口进,从出口返回,这将增大时间成本
因此,在一个函数十分简单时,可以将它写成一个宏,牺牲空间,换取时间
todo:编写简单的头文件
接下来,将通过编写一个简单的头文件来帮助熟悉#ifdef,#undef,inline等语句和关键字
编译
介绍
C语言的编译过程即将高级语言转换为汇编语言的过程,它会将所有语句转换为汇编指令,并且会优化源代码
可以用gcc生成一个编译后的文件(后缀为.s)
1 | gcc -S file.i -o file.s |
可以发现生成后的文件已经是汇编代码了
C语言语法
在这个阶段,开始涉及C语言的各种语法
数据类型
C语言有许多关键字,其中不少是用于声明变量的
1 | // 占用字节 16位 32位 64位 |
4位字节如int范围为-2147483648~2147483647,unsigned int范围为0~4294967295(20~43亿)(232)
8位字节如long long范围为-9223372036854775808~-9223372036854775807
unsigned long long范围为0~18446744073709551615(264)
除了基本的数据类型,还可以声明结构体来储存多个类型的值
1 | struct STRUCT { |
给结构体变量的分配空间遵循对齐原则
- 从第一个成员开始往下分配空间,默认从地址0开始存放
- 每个成员的偏移量必须是该成员基本数据类型的整数倍
- 总大小是最大成员基本数据类型的整数倍
例如,char占1个字节,那么它可以放在0,1,2,3,…地址上;int占4个字节,那么它可以放在0,4,8,12,…地址上
指针类型占8个字节,那么它可以放在0,8,16,…地址上;数组也同理,将它看作许多个基本类型成员的集合计算
数据间的转换
C语言的char型比较特殊,它被转换为int数据存储,并用ASCII编码,可以和int类型数据相加减
它可以用”%d”转换符输出,但并非是一个整型数据!
一般数据类型的转换如图所示,在赋值运算中可以进行自动类型转换将右值转换为左值的类型,在表达式运算中可以从低到高进行自动类型转换,例如整数相除的结果会被去掉小数部分 char → int → unsigned → long → double ← float 指针类型十分严格,如果类型不同,则不能进行互相转换
void*无类型指针是特例,可以强制转换为任意类型的一级指针,它是抽象的,并不具有一般指针类型变量的加减或给其他变量赋值的操作,它只能是左值
数据的前缀
1 | 011 // 表示八进制 |
数据的后缀
1 | // 浮点数的指数表达法,后面的数字必须是整数[1.0或表达式(5+4)也是错误的],e或E均可 |
运算符
| 运算符 | 优先级 | 运算顺序 |
| [] | 1 | 从左到右 |
| () | ||
| 对象成员. | ||
| 指针成员-> | ||
| 单目- | 2 | 从右到左 |
| 取反~ | ||
| 自增++/自减–(包括前后缀) | ||
| 解引用* | ||
| 取地址& | ||
| ! | ||
| 强制类型转换(type) | ||
| sizeof | ||
| / | 3 | 从左到右 |
| 乘法* | ||
| % | ||
| + | 4 | |
| - | ||
| << | 5 | |
|
|
||
|
|
6 | |
= |
||
| < | ||
| <= | ||
| == | 7 | |
| != | ||
| <= | ||
| 按位& | 8 | |
| 按位^ | 9 | |
| 按位| | 10 | |
| && | 11 | |
| || | 12 | |
| ?: | 13 | 从右到左 |
| 赋值符 | 14 |
基于优先级以及结合顺序,在这里讨论一些容易混淆的语句
1 | // ++与*优先级相等,而它们是从右到左结合的 |
赋值运算符优先级极低,且赋值表达式的结果就是右值的结果
数组
数组将多个相同类型的数据连在一起,方便进行访问和管理,许多数据类型的结构都可以由数组实现
二维数组
初始化二维数组时,它的列不能为空
1 | // 一层循环遍历二维数组 |
VLA(变长数组)
VLA是一个概念,有许多种方法可以实现它,其中肯定需要用到的是<stdlib.h>里的malloc()或calloc()和free()函数(在C++中可用new和delete关键字)
引入指针后,VLA的实现便很容易理解了
1 |
|
函数
C语言是一门面向过程的编程语言,函数对于它十分重要,它能把程序划分为多块,分别实现不同的功能
接口与返回值
1 | type function(paras){statements} |
上述例子里,paras表示参数,它是进入这个函数的入口时由上一进程传入的数据,type规定了function()的返回值的类型,如果不显式声明则默认为int
形参
在函数中改变形参值无法改变上一进程中传进来的值,如要改变,可以传指针进行修改
递归
递归很适合用于处理需要逆向思考的问题,例如一包数据的倒序存储等;但递归也容易造成栈溢出,递归的层级越大,风险便越大
涉及递归函数时,首先需要找到递归出口,即递归基础,每进行一次递归,数据应越来越靠近这一出口
其次是递归步骤,需要找到每次递归的数据间的联系与规律
指针
C语言的指针强大,灵活
指针基本类型
| 数据类型 | 名称 | 意义 |
| type* | 一级指针 | 指向一般数据类型的指针 |
| type** | 二级指针 | 指向指针的指针 |
| type* | 数组指针(行指针) | 指向数组的指针 |
数组名的类型以及易混淆的声明
数组名的类型很好记,只要知道该数组装载着的最高维数据的类型,就知道数组名的类型是指向这一类型的指针
| 声明方式 | 数据类型 | 意义 |
| type a[n] | type* | 本身是一个一维数组,a是指向一般数据的常量指针 |
| type a[m][n] | type* | 本身是一个二维数组,a是指向含n个元素的数组的常量指针 |
| type *a[n] | type** | 本身是一个指针数组,a是指向指针的常量指针 |
| type*at | type* | 指向一维数组的指针,a[i]是未定义的 |
函数中的指针参数
向函数传递指针
地址或数组名向函数传递时,都会退化为指针,而且建立一个形参指针变量
函数内的形参指针拥有一般指针变量的性质:例如可以进行加减(就算传入数组名,也会新建一个形参指针)
由函数返回地址
返回形参或静态量的地址有意义
返回局部变量的地址无意义:局部变量离开函数时被销毁,故访问该地址没有意义
关于赋值
指针类型限定很严格,只能用相同类型的值赋值给指针变量
1 | void func(int, int); |
位运算
汇编
介绍
汇编即把汇编代码转换为机器语言(二进制机器码)的过程
可以用gcc生成一个汇编后的文件(后缀为.o)
1 | gcc -c file.s -o file.o |
汇编语言
介绍
汇编语言(assemble language,即asm)面向机器,将许多长串的二进制码编写为一套短小精炼的命令语句,它帮助我们在寄存器、IO出口和存贮器间传递数据;将命令变为二进制码让机器听懂等。如果读者常使用IDA,则对汇编语言会有所了解
每一种处理器都有它们的一套汇编指令,它是软件开发者和机器直接进行交互的桥梁,对理解硬件之间的联系很有帮助
高级语言中提供的十六进制与八进制,正是因为汇编语言的数据会以2的幂来存储
数据间的转换
二进制数据每进四个位对应的十六进制数将进一位,每进三个位对应的八进制数将进一位
2n进制数拥有的这种特性让计算变得规律易懂
1 | 1111-1011-1110-0101 ---> 0xF-B-E-5 |
负数的表示:二进制补码
1 | -num == ~num + 1 |
数据相减即让减数取反加1后进行加法运算,因为最后一位会因进位而丢失,所以可以正确表示两数相减
如果规定为无符号数据,则最后一位参加,最大值变为两倍(即28),否则为 − 27 ∼ 27 − 1
数据寻址
处理器一次可以访问多个字节的内存,它先从内存获取指令,然后解码,最后执行;存储数据时,从低位开始压入栈,即会将低位字节存储于低地址上;当它取出数据时,会从高位开始取,这就像先入后出的栈一样
通过这种方式,处理器搭建了一座内存和寄存器间的桥梁
NASM
NASM全称The Netwide Assembler,可以在Windows和Linux下使用
我们在Linux环境下配置NASM,输入sudo apt install nasm即可
汇编语法
汇编分为三部分:data, bss, text
1 | section.data |
data: 初始化数据或常量
bss: 声明变量
text:
保存实际代码,必须以global _start开头,它规定了执行的开始位置
;: 注释,以英文';'开头
内存段分为数据段、代码段和堆栈段,数据段由.data和.bss表示、代码段由.text表示
汇编语句
汇编有三种类型的语句,且汇编语言不区分大小写
- 可执行指令
- 汇编器指令(伪指令)
- 宏指令
第一种语句告诉处理器要做什么,每一条语句汇编后都是一组二进制数,可被CPU执行;第二种语句不会变成机器语言,它们不可执行,但为汇编程序提供汇编信息(如数据和指令的区分,或数据的字长,或数据地址等),辅助源程序的汇编;宏即一种文本替换机制,用于提高编程效率
每行只有一条语句,每条语句遵循下面的格式
1 | [label] mnemonic [operands] [;conment] |
数据项
常量:直接编码于指令中,不额外分配主存空间,也不存储在存贮器里
| 进制 | 后缀 |
| 十进制 | d/D(可省略) |
| 十六进制 | h/H,以A-F开头时需在最高位加0 |
| 二进制 | b/B |
变量:保存在可读可写的主存空间,声明一个变量的指令如下
1 | name DB n dup(2,dup(4)) ;dup为复制操作符,它会将括号内数据复制n遍,例如本例n=2: 2,4,4,2,4,4 |
定义变量的伪指令如下图
| 助记符 | 变量类型 | 所占位 |
| byte(DB) | 字节 | 8 |
| word(DW) | 字 | 16 |
| dword(DD) | 双字 | 32 |
| qword(DQ) | 四字 | 64 |
| tword(DT) | 十字 | 160 |
表达式:
| 运算符类型 | 运算符 |
| 算术运算符 | + |
| - | |
| * | |
| / | |
| mod | |
| 逻辑运算符(只能对数值进行的按位运算) | and |
| or | |
| not | |
| xor | |
| 关系运算符(只能对数值的操作)成立返回0FFFFh(-1),否则返回0000h | eq(=) |
| le(<=) | |
| ge(>=) | |
| lt(<) | |
| gt(>) | |
| ne(!=) | |
| 取值运算符 | seg(获取段地址) |
| offset(获取段内偏移地址) | |
| type(获取类型属性,返回该类型的值) | |
| length(返回变量中元素个数) | |
| size(返回所占数据区的字节总数) |
链接
介绍
之前提到预处理将展开库函数,事实上,这个过程导入的只有原型,真正的实现在链接库里
而链接即把机器码和其它文件,库文件,启动文件链接起来的过程,将会生成一个可执行文件
可以用gcc生成一个链接后的文件,已经到最后步骤了,需要做的事情很少
1 | gcc file.o -o file.out |