一、课程内容介绍
C语言、C++语言、数据结构和算法,重点关注算法逻辑,无需要关注平台。
UNIX系统、Linux系统,重点关注系统接口,需要关注代码的运行平台。
通过学习系统接口,来提高函数的设计能力,在理解操作系统的运行机制的基础上,写出功能更强、更安全、更高效的代码。
二、UNIX系统介绍
诞生于1971年美国AT&T公司的贝尔实验室,主要开发者是丹尼斯.里奇、肯.汤普逊。
系统特点:多用户、多任务、支持多种处理器架构、高安全性、高可靠性、高稳定性。
既可以构建大型关键业务系统的商业服务器,也可以构建面向移动终端、手持设备、可穿戴设备的嵌入式应用。
三、Linux系统介绍
是一款类UNIX系统,免费开源,不同的发行版使用相同的内核,一般用在手机、平板、路由器、台式计算机、大型计算机、超级计算机。
从严格意义上来说,Linux仅指的是操作系统内核,隶属于GNU工程,发明人叫 Linus Benedict Torvalds。
标志:是一只企鹅,是南极的标志性动物,而目前南极不属于任何国家,为全人类所共有的,而Linux用它来当操作系统就意味这款系统属于全人类。
Minix操作系统:荷兰的Andrew S. Tanenbaum教授所开发的一款不包含任何UNIX源码的类UNIX系统,Linus Torvalds虽然深受Minix的启发写出了第一版本的Linux内核。
GNU工程:发起于1984年,由*软件基本会提供支持,它基本原则就是共享,目的是发展出一个有别一切商业UNIX系统的,免费且开源的类UNIX系统,也叫GNU Linux,是目前全球最大的开源组织。
POSIX标准:可移植操作系统接口(英语:Portable Operating System Interface,缩写为POSIX)是IEEE为要在各种UNIX操作系统上运行软件,而定义API的一系列互相关联的标准的总称。
Linux完全遵循了POSIX标准,所以在Linux下编写的代码,经过稍微修改可移植到UNIX上。这两操作系统的API,名字相同、参数相同、返回值相同。
三、GNU编译工具
是GNU组织为了编译Linux内核源码而开发的一款编译工具,经过长时间的发展目前已经成为一个编译平台,能够支持多种编程语言(C、C++、Java、Objective-C、Ada、C#)和操作系统(UNIX、Linux、Windows),gcc -v 可以查看编译器版本。
构建(Build)过程:预处理、编译、汇编、链接
gcc -E code.c -o code.i -> code.i
gcc -S code.i -> code.s
gcc -c code.s -> code.o
gcc code.o ... -> a.out
文件后缀:
.h 头文件
.h.gch 头文件的编译结果,会被优先使用,建议立即删除
.c 源文件
.i 预处理文件
.s 汇编文件
.o 目标文件
.a 静态库文件
.so 共享(动态)库文件
参数:
-E 预处理
-S 产生汇编文件
-c 只编译不连接
-o 指定输出文件的名字
-Wall 尽可能多的产生警告
-Werror 把警告当错误处理
-x 指定源代码的语言
-g 生成调试信息
-I 指定头文件的加载路径
-L 指定库文件的加载路径
-lname 要额外加载
-pedantic 对不符全ANSI标准的语法产生警告(对gcc扩展的语法产生警告)。
编译多个文件:
1、头文件作用
头文件卫士
声明外部变量、函数、类、结构、联合、枚举
定义宏、类型别名
包含其它头文件
2、包含头文件时要注意
#include <> 从系统指定的位置加载。
#include "" 先从当前目录下加载,如果当前目录没有再从系统指定的位置加载。
当没函数声明时,编译器会猜测函数的格式,这种猜测行为叫函数隐式声明。
编译器支持的预处理指令:
#include 导入头文件
#define 定义宏常量、宏函数
## 连接两个标识符
# 把参数转换成字符串字面值
#undef 删除宏
#if 判定
#ifdef/#ifndef 宏定义判定
#else 与#if/#ifdef/#ifndef配合使用
#elif 相当于else if
#endif 判定的结束标志
#error 产生错误
#warning 产生警告
#line 修改当前行号
#pragma GCC dependency "文件名" 监控某个文件不被私自修改
#pragma GCC poison <标识> 把某个标识符当病毒处理,监控某个标识符不被使用。
#pragma pack(1/2/4) 修改结构、联合的补齐、对齐字节数
预定义的宏:
__FILE__ 文件名
__func__ 函数名
__LINE__ 行号
__DATA__ 日期
__TIME__ 时间
头文件的三种定位方式:
1、#include "路径/xxx.h" 直接把路径写在代码中,但如果头文件路径发生变化,需要修改源代码。
2、gcc -I目录 在编译时指定加载头文件的路径。
3、设置C_INCLUDE_PATH环境变量,来添加头文件的查找路径。
打开配置文件:vim ~/.bashrc
在文件末尾添加一行内容:export C_INCLUDE_PATH=$C_INCLUDE_PATH":/home/zhizhen/Linux环境编程day01/include"
重新加载配置文件:source ~/.bashrc
env命令可以查看所有环境变量。
扩展:PATH环境变量代表是命令的查找路径,把.添加进去以后再执行程序就不需要./了。
注意:删除环境变量时需要退出终端,再重新打开
四、库
库:也叫代码库,把一个些目标文件合并在一起方便使用,有静态库和共享库两种。
静态库:在链接时把库的二进制指令复制到调用模块中。
共享库:会和调用者一起加载到内存,当执行调用语句时会从程序的调用位置跳转到共享库中运行。
优缺点:
静态库的优点是运行速度快,但维护麻烦,当静态库中人内容更新后需要重新编译程序,使用静态库编译出的可执行文件会比共享库的要大。
共享库的优点是使用方便,共享库如果发生变化不需要重新编译程序,使用它编译出的可执行文件比使用静态库要小,运行速度要比使用静态库慢。
注意:静态库的扩展名是.a,共享库的扩展名是.so,共享库要有执行权限。
五、静态库的创建与使用
1、创建静态库
编辑静态库原码:vim .c/.h
编译出目标文件:gcc -c xxx.c -> xxx.o
把目标文件打包成静态库文件:ar -r libxxx.a x1.o x2.o ...
ar 是一个专门控制静态库的命令
-r 把目录文件合并成一个静态库,如果静态库文件已经存在则更新。
-q 向静态库中添加目录文件
-t 查看静态库中有哪些目标文件
-d 从静态库中删除目标文件
-x 把静态库展开为目标文件
2、使用静态库
1、直接调用
把共享库当作目标文件一样,与调用者的目标文件一起合并出可执行文件。
gcc main.c libcalc.a
2、通过设置LIBRARY_PATH环境变量来指定库的路径
gcc main.c -lcalc 需要通过-l来指定库名
3、通过gcc -L参数来指定库的路径
gcc main.c -L./calc/ -lclac
六、共享库的创建与使用
1、创建共享库
编辑静态库原码:vim .c/.h
编译出目标文件:gcc -c -fpic xxx.c -> xxx.ols
把目标文件打包成共享库:gcc -shared x1.o x2.o ... -o libxxx.so
-fpic编译出位置无关代码,在代码中使用相对地址,这样共享库就可以遇到内存的任何位置。
2、使用共享库
1、直接调用
gcc main.c libcalc.so
注意:需要设置共享的加载路径,LD_LIBRARY_PATH
2、通过设置LIBRARY_PATH环境变量来指定库的路径
gcc main.c -lcalc 需要通过-l来指定库名
注意:如果静态库和共享库同时存在,优先使用共享库,通过-static可以指定使用静态库。
3、通过gcc -L参数来指定库的路径
gcc main.c -L./calc/ -lclac
4、动态加载共享库
#include <dlfcn.h>
void *dlopen(const char *filename, int flag);
功能:打开共享库
filename:共享库的路径
flag:打开方式
RTLD_LAZY:延迟加载,使用到共享库时再加载
RTLD_NOW:立即加载
返回值:成功返回共享库的句柄,失败返回NULL
void *dlsym(void *handle, const char *symbol);
功能:通过函数名在共享库中获取函数指针
handle:共享库的句柄
symbol:函数名
返回值:函数地址,失败返回NULL
char *dlerror(void);
功能:获取错误信息
int dlclose(void *handle);
功能:卸载共享库
注意:编译时添加-ldl参数
七、辅助工具
nm:查看目标文件、可执行文件、静态库、共享库文件的符号列表。
ldd:查看可执行文件依赖了哪些共享库
strip:删除目标文件、可执行文件、静态库、共享库文件中符号列表、调试信息,可以有效降低文件的大小。
objdump:可以显示二进制文件的汇编信息
八、错误处理
1、通过函数返回值表式错误
1、返回值合法表示成功,非法表示失败。
2、返回有效指针表示成功,返回NULL/0xFFFFFFFF 表示失败。
3、成功返回0,失败返回-1。
4、永远成功,如:printf
练习:
int str_len(const char* str); 求字符串长度,如果空指针则报错。
char* str_cpy(char* dest,const char* src): 字符串拷贝,如果目标位置无效则报错。
int min(int num1,int num2,int* p); 求两个整数的最小值,如果二者是相等则报销错。
int avg(int num1,int num2); 求两个整数的平均值,考虑溢出问题,该函数永远成功。
2、通过errno表示错误,系统的API出错时会影响errno的值。
可以根据errno的错误编号来获取错误原因
1、strerror 根据errno错误转换字符串信息
2、printf %m 同上
3、perror 同上
注意:errno是一个全局的变量,需要通过函数的返回值判断函数执行是否出错误,再通过errno获取错误原因。
如果API执行成功并不会修改errnor值,所以不能通过errno的值判断函数执行是否成功。
九、环境变量表
每个程序执行时都收到一张操作系统传递给的环境变量表,方便该程序了解当系统的环境配置,env命令可以在终端查看环境变量表。
环境变量表是一个以NULL结尾的字符指针数数组。
该表是系统环境变量的拷贝,程序对当前这张进行修改,并不影响制作系统。
char *getenv(const char *name);
功能:根据环境变量名获取环境变量的值
int putenv(char *string);
功能:name=value向环境变量表中添加环境变量,环境变量不存在则添加,存在则修改。
返回值:0表示成功,非零表示失败
注意:不会把字符串拷贝到环境变量表中,而是把字符串地址存储到数组中。
int setenv(const char *name, const char *value, int overwrite);
overwrite:
零 环境变量已经存在则不修改
非零 功能相当于putenv
返回值:0表示成功,非零表示失败
int unsetenv(const char *name);
功能:删除环境变量
返回值:0表示成功,非零表示失败
int clearenv(void);
功能:清空环境变量
练习2:
1、给C_INCLUDE_PATH添加一个路径(/home/zhizhen/include)。
2、修改USER的值为ubuntu
3、删除XDG_DATA_DIRS最后一个路径
把常用的栈、队列、链表、二分查找、快速排序封装到一个静态库文件中。