Linux系统编程 day02 vim、gcc、库的制作与使用
- 01. vim
- 0101. 命令模式下的操作
- 0102. 切换到文本输入模式
- 0103. 末行模式下的操作
- 0104. vim的配置文件
- 02. gcc
- 03. 库的制作与使用
- 0301. 静态库的制作与使用
- 0302. 动态库(共享库)的制作与使用
01. vim
vim是一个编辑器,需要自行安装,可以使用命令sudo apt-get install vim。安装完之后可以在shell中输入vimtutor命令查看帮助文档。
vim有三种工作模式,分别是命令模式、文本输入模式以及末行模式。三者的切换关系如下:

0101. 命令模式下的操作
用户按下ESC键,就可以使vim进入命令模式。当用户使用vim打开一个新文件的时候也是默认进入的命令模式。命令模式中的操作如下。
| 快捷键 | 操作 |
|---|---|
| ZZ | 保存退出 |
| gg=G | 格式化代码 |
| h | 光标左移 |
| j | 光标下移 |
| k | 光标上移 |
| l | 光标右移 |
| w | 光标移动一个单词 |
| gg | 光标移动到文件开头 |
| G | 光标移动到文件末尾 |
| 0 | 光标移动到行首 |
| $ | 光标移动到行尾 |
| nG | 光标移动到第n行 |
| x | 删除光标后一个字符,相当于Delete |
| X | 删除光标前一个字符,相当于Backspace |
| dw | 删除光标后一个字单词 |
| d0 | 删除光标前本行的所有内容,不包括光标所在字符 |
| d$ | 删除光标后本行的所有内容,包括光标所在字符 |
| D | 删除光标后本行的所有内容,包括光标所在字符 |
| dd | 删除光标所在行的内容,实际上是剪切 |
| ndd | 删除光标当前行向下n行的内容,包括本行一共n行,实际上是剪切 |
| v | 非列模式视图,使用hjkl进行移动选择内容,按d删除 |
| ctrl + v | 列模式视图,使用hjkl移动选择内容,按d删除 |
| u | 撤销 |
| ctrl + r | 反撤销 |
| yy | 复制当前行的内容 |
| nyy | 复制当前行向下n行,包括本行一共n行 |
| p | 在光标位置向下开辟一行并粘贴 |
| P | 在光标位置向上开辟一行并粘贴 |
| r | 替换当前字符 |
| R | 替换当前行光标后的字符 |
| / | /xxx 从光标所在位置开始搜索,按n向下搜索,按N向上搜索 |
| ? | ?xxx 从光标所在位置开始搜索,按n向上搜索,按N向下搜索 |
| # | 将光标移动到待搜索的字符串上,然后按n向上搜索,按N向下搜索 |
| shift + k | 在带搜索的字符串上按下shift + k 或者 K 可以查看相关的帮助文档 |
| ctrl + u | 向下翻半屏,光标向上移动 |
| ctrl + d | 向下翻半屏,光标向下移动 |
| ctrl + f | 向前翻一屏 |
| ctrl + b | 向后翻一屏 |
0102. 切换到文本输入模式
从命令模式切换到文本输入模式的操作如下。
| 快捷键 | 操作 |
|---|---|
| i | 在光标前插入 |
| a | 在光标后插入 |
| I | 在光标所在行的行首插入 |
| A | 在光标所在行的行尾插入 |
| o | 在光标所在行的下面开辟一行,行首插入 |
| O | 在光标所在行的上面开辟一行,行首插入 |
| s | 删除光标后的字符并在光标当前位置插入 |
| S | 删除光标所在当前行并在行首插入 |
| 列模式插入 | 按I或者shift + i 向前插入,插入字符串后按两次ESC保存。 |
0103. 末行模式下的操作
从命名模式切换到末行模式,只需要输入:即可。下面是关于末行模式的一些操作。
| 快捷键 | 操作 |
|---|---|
| q | 退出 |
| q! | 强制退出,不保存修改内容 |
| w | 保存修改内容,不退出 |
| wq | 保存并退出 |
| x | 保存并退出 |
| : s/old/new/ | 光标所在行的第一个old替换为new |
| : s/old/new/g | 光标所在行的所有old替换为new |
| :m,ns/old/new/g | 将第m行到第n行的所有old替换为new |
| :%s/old/new/g | 将当前文件的所有old替换为new |
| :1,$s/old/new/g | 将当前文件的所有old替换为new |
| :%s/old/new/gc | 将当前文件的所有old替换为new,但是每次替换需要用户确认 |
| !shell命令 | 执行shell命令,按下两次ESC可以回到命令模式 |
| sp | 当前文件水平分屏 |
| vsp | 当前文件垂直分屏 |
| sp 文件名 | 当前文件和另一个文件水平分屏 |
| vsp 文件名 | 当前文件和另一个文件垂直分屏 |
| ctrl + w + w | 多个窗口间切换光标 |
| wall | 保存所有分屏窗口 |
| wqall | 保存并退出所有分屏窗口 |
| xall | 保存并退出所有分屏窗口 |
| qall | 退出所有分屏窗口 |
| qall! | 强制退出所有分屏窗口 |
分屏模式可以在shell中使用以下命令:
# 水平分屏
vim -on 文件1 文件2 ....
vim -o 文件1 文件2 ....
# 垂直分屏
vim -On 文件1 文件2 ....
vim -O 文件1 文件2 ....
0104. vim的配置文件
vim的配置文件分别是系统级别配置文件/etc/vim/vimrc和用户级别配置文件~/.vimrc。修改系统级别的配置文件会影响到该系统下的所有用户,如果修改的是用户级别的配置文件只会影响到当前的用户而不会影响到其它用户。在配置文件中,我们可以添加以下的设置。
| 命令 | 功能 |
|---|---|
| set nu | 设置显示行号 |
| set nonu | 设置不显示行号 |
| set shiftwidth = n | 设置gg=G缩进为n个空格,默认是8个空格 |
| set autoindent | 设置自动换行,每行按与上一行同样的标准进行缩进 |
| set list | 将制表符显示为^|,用$表示行尾 |
| set showmatch | 在vim中输入相应的右括号时光标会暂时回到相匹配的左括号 |
| set tabstop = n | 设置tab缩进的字符数目为n |
| set encoding=utf-8 | 设置编码格式为utf-8 |
| set smartindent | 为C程序提供自动缩进 |
这些命令也可以在末行模式下使用。
02. gcc
gcc编译器将C语言原程序文件生成一个可执行文件一共需要经理四个步骤,分别是预处理、编译、汇编、链接。


这四个步骤的命令如下:
# 预处理
gcc -E hello.c -o hello.i
# 编译
gcc -S hello.i -o hello.s
# 汇编
gcc -c hello.s -o hello.o
# 链接
gcc hello.o -o hello
当然我们可以一步到位:
gcc hello.c -o hello
接下来我们来看一看关于gcc的一些常用参数
| 参数 | 功能作用 |
|---|---|
| -v | 查看gcc版本号 |
| –version | 查看gcc版本号 |
| -E | 预处理,生成预处理文件 |
| -S | 编译,生成汇编文件 |
| -c | 只编译,生成.o文件,通常称为目标文件 |
| -I | 指定头文件所在路径 |
| -L | 指定库文件所在路径 |
| -l | 指定库名称 |
| -o | 指定生成的目标的名字 |
| -g | 包含调试信息,使用gdb调试的时候需要添加-g参数 |
| -On n=0~3 | 编译优化,n的值越大优化得越多 |
| -Wall | 提示更多警告信息 |
| -D | 编译时定义宏 |
03. 库的制作与使用
库是二进制文件,是源代码文件的另一种表现形式,可以认为是加了密的源代码。库是一些功能相近或者相似的函数的集合体。库可以提高代码的重用性,提高程序的健壮性,同时也可以减少开发者的代码量,缩短开发周期。库可以分为静态库和动态库,动态库也被称为共享库。
0301. 静态库的制作与使用
静态库可以认为是一些目标代码的集合,是在可执行程序运行前就已经加入到执行代码中,称为可执行程序的一部分。静态库我们习惯于用.a作为后缀名,前缀为lib,库的名称是自定义的。比如有一个库的名称是test,则该静态库的库文件名就是libtest.a。
静态库的制作主要分为两步,第一步是使用gcc中的-c参数将C语言源程序文件编译成.o文件。第二步是使用打包工具ar将.o问加你打包为.a文件。
例如此处我们有add.c、sub.c、mul.c、dive.c、main.c、head.h五个文件。其中head.h中为add.c、sub.c、mul.c、dive.c的声明部分,实现在各自的.c文件中。接下来我们需要将前面四个.c文件进行打包制作静态库,库名为smath。
首先将他们编译成.o文件。
# 第一种方法是全部直接编译
gcc -c add.c sub.c mul.c dive.c
# 第二种方法是逐个编译
gcc -c add.c -o add.o
gcc -c sub.c -o sub.o
gcc -c mul.c -o mul.o
gcc -c dive.c -o dive.o
第二步使用ar打包工具进行打包为静态库文件。
# 将.o文件打包成静态库文件
ar rcs libsmath.a add.o sub.o mul.o dive.o
在打包命令中,参数r表示更新、c表示创建、s表示建立索引。
如此以来就创建好了一个静态库文件。那么怎么使用呢,我们可以通过gcc的参数去指定静态库文件以及头文件的所在路径即可。需要注意的是-l指定库名称的时候不需要写库的前缀lib以及后缀.a。如libsmath.a只需要写smath即可。另外-L、-l、-I参数后面跟的参数可以用空格隔开,也可以不隔开。如接下里我们需要编译main.c并执行。
# 生成可执行文件,指定头文件为当前路径,库文件为当前路径,库名为smath
gcc main.c -I./ -L./ -lsmath -o main
静态库文件制作好了之后,需要将.a文件和头文件一并发布给用户。静态库最终打包到应用程序中,实现函数本地化,寻址方便、速度快。程序在运行时与函数库再无任何关系,移植方便。但是静态库消耗系统资源较大,每一个进程使用静态库都需要复制一份,浪费内存。静态库会给程序的更新、部署和发布带来麻烦。如果静态库更新了,那么所有使用它的应用程序都需要重新编译、发布给用户。对于使用者来说,也就是一个很小的改动都需要重新下载整个程序。
0302. 动态库(共享库)的制作与使用
共享库的出现正好解决了上面动态库的不足。共享库在程序编译时并不会被连接到目标代码中,而是在程序运行的时候才会被载入。不同的应用程序如果调用相同的共享库,那么在内存中只需要有一份该共享库的拷贝即可,避免了内存空间浪费的问题。由于动态库是程序运行时才会被载入,也解决了静态库对程序的更新、部署和发布带来的麻烦。动态库习惯于以.so作为后缀名,前缀也需要加上lib。比如有个动态库的名字为dmath,则完成的动态库文件名为libdmath.so。
动态库的制作也有两个步骤,第一个步骤是使用gcc中的-fpic参数创建与地址无关的编译程序,生成目标文件.o。第二步是使用gcc的-shared生成动态库。
同样,我们以静态库的那个例子生成动态库,第一步生成目标文件的过程如下:
# 生成目标文件 -fpic 创建与地址无关的编译程序 -c 只编译
gcc -fpic -c add.c sub.c mul.c dive.c
第二步是生成动态库文件,这里取库名为dmath。
# 生成动态库文件 -shared 指定生成动态库
gcc -shared add.o sub.o mul.o dive.o -o libdmath.so
如此以来就生成了动态库文件libdmath.so。那么我们如何查看一个库中有哪些内容呢?我们可以使用nm命令。
# 查看dmath动态库中有哪些内容
nm libdmath.so
# 查看smath静态库中有哪些内容
nm libsmath.a
# 查看add.o中有哪些内容
nm add.o
动态库的使用和静态库的使用方法是一样的,需要指定头文件路径、库路径以及库名。如接下来编译main.c生成可执行文件如下:
gcc main.c -I./ -L./ -ldmath -o main
这样就可以生成可执行程序main。但是在执行运行main程序的时候会报错,这是为什么呢?因为系统加载可执行代码的时候,能够直到其以来的库文件名字,但是还需要知道所依赖的库的绝对路径,此时就会有系统动态载入器,也就是lld。执行ldd命令可以查看可执行文件依赖的库文件。如下面查看main所依赖的库文件。
# 查看main所依赖的库文件
ldd main
不过执行这句话的时候有时会反馈找不到。对于ELF格式的可执行程序,是由ld-linux.so*来完成的,它搜索ELF文件的DT_RPATH段,再搜索环境变量LD_LIBRARY_PATH,再搜索/etc/ld.so.cache文件列表,最后搜索/lib/、/usr/lib/目录找到库文件并将其载入内存。如果上面的都找不到就会报错。我们可以使用file命令查看文件的类型,如查看main的类型。
# 查看main的文件类型
file main
那么如何让系统找到自己的共享库呢?首先分析程序去搜索ELF文件的DT_RPATH段,这个部分是我们不能更改的,所以我们只能对剩下的三个路径进行下手。方法如下:
- 第一种方法,拷贝自己制作的动态库放在
/lib/或者/usr/lib/目录下。 - 第二种方法, 临时设置
LD_LIBRARY_PATH,在shell中执行export LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径。 - 第三种方法,永久设置
LD_LIBRARY_PATH,将LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径设置到~/.bashrc中,然后再使环境变量生效,生效可以任使用以下的其中一个方法。- 执行
. ~/.bashrc使配置文件生效 - 执行
source ~/.bashrc使配置文件生效 - 退出当前终端,然后再次登录也可以使配置文件生效
- 执行
- 第四种方法,永久设置,把
LD_LIBRARY_PATH=$LD_LIBRARY_PATH:库路径设置到/etc/profile文件中。 - 第五种方法,添加到
/etc/ld.so.cache文件中。编辑/etc/ld.so.conf文件,假如库文件所在的目录路径,再运行sudo ldconfig -v重建/etc/ld.so.cache文件。
通过上述的方法之后,再执行ldd main就可以找到对应依赖的库文件。也就是这个程序终于可以正常运行了。
动态库把对一些库函数的链接载入推迟到了程序运行的时期。动态库实现了进程之间的资源共享。动态库使程序升级变得简单。动态库使得链接载入完全由程序员在代码中控制。
最后对比以下动态库与静态库的优缺点。
| - | 优点 | 缺点 |
|---|---|---|
| 动态库 | 1. 节省内存 2. 部署升级更新方便,只需替换动态库即可,然后再重启服务 | 1. 加载速度比静态库慢 2. 移植性差,需要把所有用到的动态库都移植 |
| 静态库 | 1. 执行速度快,因为静态库已经编译到可执行文件内部 2. 移植方便,不依赖其它库文件 | 1. 耗费内存,因为每一个静态库可执行程序都会加载一次 2. 部署更新麻烦,因为静态库修改以后所有调用这个静态库的可执行文件都需要重新编译。 |