UNIX系统简介:
1970年于美国贝尔实验室,作者肯.汤普逊和丹尼斯.里奇
UNIX是最早的多用户、多任务、支持多种CPU架构,高稳定性、高可靠性、高安全性
既能构建大型关键型业务系统的服务器(银行、电信公司等),也能支持移动嵌入式设备
Minix是一种开源的基于微内核架构的类UNIX计算机操作系统,Linux之父林纳斯.托瓦兹受到Minix的启发,开发了Linux内核的第一个版本
Linux系统简介:
Linux,全称 GNU/Linux
于1991年发布,它主要受到Minix和Unix思想的启发,是一个基于POSIX的多用户、多任务、支持多线程和多CPU的操作系统。它支持32位和64位硬件,能运行主要的Unix工具软件、应用程序和网络协议。
Linux继承了Unix以网络为核心的设计思想,是一个性能稳定的多用户网络操作系统。Linux有上百种不同的发行版
相关知识:
Linux的标志:企鹅,因为企鹅是南极洲的标志动物,而南极目前没有被任何国家占用,属于全人类
GNU计划\组织:目前全球最大的开发开源系统的开源组织,负责Linux内核源的升级和维护
GPL通用许可证:在带有GPL证书的代码基础下开发出来的软件,也都需要支持GPL证书
POSIX统一可移植操作系统接口:Linux是完全遵循,UNIX绝大部分遵循,所以它们的命令、API接口基本上都是通用的
发行版:Linux只是内核,内核+Shell+基础软件 = 用户操作系统
不同的公司可以根据Linux内核制作出不同版本、不同风格的Linux发行版,例如:Ubuntu、Redhat、CentOS、Debian
GNU编译工具gcc\g++:
多样化:
支持多种编程语言、支持各种操作系统
gcc -v 查看版本信息
编译构建过程:
预处理:gcc -E code.c -o code.i
编译:gcc -S code.i -> code.s
汇编:gcc -c code.s -> code.o
链接:gcc a.o b.o ... -> a.out
文件类型:
.c 源文件
.h 头文件
.h.gch 头文件的编译结果,会被优先使用
.i 预处理文件
.s 汇编文件
.o 目标文件(二进制)
.a 静态库文件
.so 共享库文件
编译参数:
-E 只预处理
-S 只生成汇编代码
-o 指定编译结果文件名
-c 只编译不链接
-std 指定语法标准 -std=gnu99
-g 生成调试信息
-Wall 尽可能多地产生警告
-Werror 把警告当错误处理
-L 指定库文件的查找路径
-l 指定要一起加载的库文件名字 -lm
-I 指定头文件的加载路径 <> ""
-pedantic -ansi 对于不符合ANSI/ISO标准的语法产生警告
预处理指令:
#include <>/"" 导入头文件
#define 定义宏常量、函数
#undef 删除宏
#ifdef 宏存在则条件为真
#ifndef 宏不存在则条件为真
#endif
#if/#elif/#else/#endif 条件判断编译
#error 提示错误,并阻止生成可执行文件,一般与条件判断配合
#error "提示信息"
#warning 提示警告
#line 设置行号 影响了提示错误、提示警告的行号
#pragma pack(n) 设置对齐、补齐的最大字节数
#pragma once 相当于头文件卫士 防止头文件重复导入
#pragma GCC dependency "file.h" 监控文件是否有修改
#pragma GCC poison key 设置关键字key为毒药,禁止在后续代码中使用
库:
库文件是目标文件的集合,可以被其他代码调用,把代码封装成库文件后方便使用、方便管理,安全性高、保密性强
静态库:
就是目标文件的集合,当调用静态库时,编译器会把静态库的所有的二进制指令拷贝到最后的可执行文件中
优点:运行速度比共享库要快,运行时不需要依赖静态库文件
缺点:可执行文件相对较大,当静态库有修改后,所有使用了该静态库的可执行文件都需要重新编译
共享库:
就是带入口的可执行文件,当调用共享库时,调用语句处会记录该函数在共享库中的位置,并且共享库文件要与运行的可执行文件一起加载到内存中,当执行到调用共享库代码的语句时,就会跳转到共享库二进制指令的内存中执行,执行完后会跳转回调用语句处
优点:可执行文件相对较小,当共享库有修改,可执行文件不需要重新编译
缺点:运行速度比静态库慢一些,运行时需要依赖于共享库文件
静态库:
制作静态库:
1、编译出目标文件
gcc -c xxx.c ...
2、打包目标文件生成静态库文件
ar -r libxxx.a a.o b.o c.o...
使用静态库:
1、直接使用
gcc xxx.c libxxx.a
2、指定库文件的位置
gcc xxx.c -Lpath -lxxx
-L指定库的路径 -l库名 指定要加载的库(库名不包括前缀lib 后缀.a .so)
3、通过修改配置文件中的环境变量来指定库的查找路径
打开系统配置文件:
vim ~/.bashrc
在配置文件末尾添加:
export LIBRARY_PATH=$LIBRARY_PATH:新路径(绝对路径)
保存退出并重新加载配置文件
source ~/.bashrc
使用静态库:
gcc xxx.c -lxxx
共享库:
制作共享库:
1、编译生成目标文件
gcc -fpic -c xxx.c
-fpic 生成与位置无关的文件
2、生成共享库文件
gcc -shared -fpic a.o b.o c.o ... -o libxxx.so
使用共享库:
编译时:
1、直接使用
gcc xxx.c libxxx.so
2、指定库文件的位置
gcc xxx.c -Lpath -lxxx
-L指定库的路径 -l库名 指定要加载的库(库名不包括前缀lib 后缀.a .so)
3、通过修改配置文件中的环境变量来指定库的查找路径
打开系统配置文件:
vim ~/.bashrc
在配置文件末尾添加:
export LIBRARY_PATH= $ LIBRARY_PATH:新路径(绝对路径)
保存退出并重新加载配置文件
source ~/.bashrc
使用共享库:
gcc xxx.c -lxxx
运行时:
运行时,系统会去默认指定的共享库加载路径进行加载共享库,一般通过修改配置文件的环境变量进行设置
export LD_LIBRARY_PATH=$ LD_LIBRARY_PATH:共享库的加载路径
注意:
1、如果指定路径下有同名的静态库文件、共享库文件,gcc编译器会默认优先加载共享库文件
如果想要优先使用同名静态库文件 增加 -static编译参数实现
2、对配置文件中的环境变量进行删除修改时,最好重启终端才生效
练习:
把常用的数据结构、算法封装成libdsa.so
把常用的头文件放入 /usr/include
栈、队列、通用链表、平衡二叉树、顺序查找、二分查找、快排、冒泡、计数、插入、选择
动态使用共享库:
作用:在代码中决定使用哪些共享库,编译时不需要指定,执行时还是需要依赖共享库
dlopen\dlerror\dlsym
库文件相关辅助命令:
ldd 查看可执行文件依赖哪些共享库 ldd ./a.out
nm 查看目标文件、可执行文件、静态库、共享库文件的符号列表
strip 减肥 删除目标文件、可执行文件、静态库、共享库文件的符号列表
环境变量表:
操作系统会给每个执行的程序提供一张环境变量表,该表中记录了操作系统在提供的时候所处于的所有环境变量,这些环境变量反映了操作系统的配置情况以及该程序所处操作系统中的环境情况
通过声明 extern char** environ; 就可以使用该程序的环境变量表
这个表可以随意使用,因为它只是操作系统提供的一张拷贝表而已
extern char** environ;
for(int i=0; environ && environ[i]; i++){
printf("%s\n",environ[i]);
}
操作环境变量表的函数:
char *getenv(const char *name);
功能:通过环境变量名获取它的值
int setenv(const char *name,const char *value,int overwrite);
功能:向环境变量表中添加新的环境变量
name:环境变量名 value:环境变量的值
overwrite: 当环境变量存在时 非0 修改原来环境变量的值 0 不修改
返回值:成功返回0 失败返回-1
int putenv(char *string);
功能:向环境变量表中添加新的环境变量或修改值 string:以 环境变量名=环境变量值
int unsetenv(const char *name);
功能:删除环境变量 返回值:成功返回0 失败返回-1
int clearenv(void);
功能:清空环境变量表
#include <stdio.h>
#include <stdlib.h>
extern char** environ;
int main(int argc,const char* argv[])
{
for(int i=0; environ[i]; i++)
{
printf("%s\n",environ[i]);
}
printf("-----%s----\n",getenv("LIBRARY_PATH"));//通过环境变量名获取它的值
printf("%d\n",setenv("HEHE","hehe",0));//向环境变量表中添加新的环境变量
printf("-----%s----\n",getenv("HEHE"));
printf("%d\n",setenv("HEHE","xixi",1));
printf("-----%s----\n",getenv("HEHE"));
clearenv();//清空环境变量表
printf("%p\n",environ);
for(int i=0; environ && environ[i]; i++)
{
printf("%s\n",environ[i]);
}
}
错误处理:
1、通过函数返回值表示出现错误
a、合法值表示成功、非法值表示失败
例如:查找位置、计算大小等
b、返回值是指针类型时,返回NULL或者0xFFFFFFFF表示失败,其余都表示成功
malloc\mmap
c、返回0表示成功,-1表示失败,一般都是操作系统函数
d、永远成功 例如:printf
2、通过改变全局的错误编号,来表示出现了错误
errno 定义在 errno.h中
由于系统函数执行导致出现错误,就会改变errno的值
char *strerror(int errnum);
功能:根据errno的值获取对应的错误提示信息
注意:errno是一个全局变量,所以不能只根据它的值是否为0就简单地判断是否产生错误