c/c++程序的编译过程(GCC的编译流程)
主要分为四个阶段:预处理,编译阶段,汇编阶段,链接阶段
当我们写好一个C语言程序的时候,其文件的后缀为.c,称它为源程序。
预处理:编写好的源程序通过预处理器,预处理器会识别字符#开头的命令,告诉预处理器进行读取字符#后对应的系统头文件内容,并将这些读取的内容插入到程序文本中(宏替换),得到另一个比之前大的C程序,此时修改后的文件的后缀名通常为.i。
编译阶段:编译器(ccl)会将预处理过的C程序进行编译,得到一个汇编语言程序(此时的文件还是文本文件,文件后缀为.s),使我们写的源程序变成更加接近低级机器语言指令。
汇编阶段:汇编器(as)的工作就是将编译阶段得到的汇编语言程序翻译成机器语言指令,并把这些指令打包成可重定位目标程序的格式,打包完毕后将其保存在目标文件中,该文件的后缀为.o,此时的文件是二进制文件。
链接阶段:汇编阶段得到的目标文件还无法直接运行,因为源程序中有可能有多个文件,而汇编阶段做的仅仅只是将每个文件进行处理,所以链接器(ld)就是将各个目标文件进行合并处理,使他们变成一个可以被加载到系统内存中执行的统一整体,这时候称它为可执行文件。
强类型语言 VS 弱类型语言
- 强类型语言是一种强制类型定义的语言,一旦某一个变量被定义类型,如果不经过强制转换,则它永远就是该数据类型了,强类型语言包括Java、.net 、Python、C++等语言。
- 弱类型语言是一种弱类型定义的语言,某一个变量被定义类型,该变量可以根据环境变化自动进行转换,不需要经过显性强制转换。
- 无论是强类型语言还是弱类型语言,判别的根本是是否会隐性的进行语言类型转变。强类型语言在速度上略逊于弱类型语言,但是强类型定义语言带来的严谨性又能避免不必要的错误。
数组和指针的区别
指针 | 数组 |
---|---|
存放的是数据在内存中的地址 | 存储多个相同类型数据的集合 |
同类型指针可以相互赋值 | 数组不可以,只能一个一个元素赋值、拷贝 |
指针是间接访问数据,需要通过解引用来获得指定地址的内容 | 数组可以直接通过下标访问数据 |
sizeof(指针)是指针变量的大小 | sizeof(数组名)数组所占存储空间的内存 |
指针名可以表示地址,并且可以修改 | 数组名也可以表示地址但是不可以修改 |
补充:数组传参时会退化成为指针
C语言只会以值拷贝的方式传递参数,参数传递时,如果只拷贝整个数组,效率会大大降低,并且在参数位于栈上,太大的数组拷贝将会导致栈溢出。因此,C语言将数组的传参进行了退化。将整个数组拷贝一份传入函数时,将数组名看做常量指针,传数组首元素的地址。
字符数组和字符串常量
const char * arr = "123";//字符串常量
字符串123保存在常量区,const本来的作用使arr指向的值不能通过arr来修改,但是字符串123保存在常量区,本来就不可修改,所以加不加const效果都是一样的
char * brr = "123"; //字符串常量
字符串123保存在常量区,这个arr指针指向的是同一个位置,同样不能通过brr去修改123的值
const char crr[] = "123";
这里123本来是在栈上的,但是编译器可能会做某些优化,将其放到常量区
char drr[] = "123";//字符数组
字符串123保存在栈区,可以通过drr进行修改
此处的赋值是将常量区的字符串"123"拷贝到了堆栈区的数drr的空间了。数组drr是在堆栈区开辟了空间,此时是可以修改字符串的值,因为修改的是堆栈区的字符串的值。另外此时的数组名drr是堆栈区中的"hello"的首地址。
static关键字的作用
c++和c的区别
c++中的四种cast
c++中指针和引用的区别
指针和数组的区别
野指针&悬空指针
虚函数&纯虚函数
为什么析构函数是虚函数&为什么C++默认的析构函数不是虚函数
函数指针VS指针函数
C++中析构函数的作用
静态函数与虚函数的区别
重载&重写&覆盖
多态:静态多态&动态多态
i++与++i的区别与实现
四种智能指针的实现
字符串之间的区别