本章将介绍Linux下常用工具的安装及使用,在Linux下安装软件,编写代码,调试代码,以及和远程gitee仓库交互等操作。
在centOS 7中安装软件:
其中源码安装和rpm安装并不简单,当依赖别的库时,还需要下载其他的库,Windows是直接打包好了的。
yum安装的好处:不用编译源码,不用解决软件的依赖关系。
当我们要安装别人的软件:
我们日常生活中就是上述的方式安装软件的,当然Liunx安装软件也是这样的。
yum类似于手机上的应用市场APP、迅雷。
yum list:
我们推荐先下载一下两个软件:
yum list | grep lrzsz
yum install -y lrzsz.x86_64
yum install -y epel-release
注意:我们下载软件要用root身份,或者通过sudo来提升权限。
如何知道去哪台服务器上下载软件呢?
/etc/yum.repos.d
去找对应链接。有的时候在下载的时候会发现,下载的速度非常慢,这是因为有的yum源不是在国内,而是在国外的。这时候就需要我们配制国内的yum源了。
注意:做任何配置,绝对不要先删除,一定是先备份(就是将之前的目录改个名字)。
vim是什么?
只能用来写代码,功能强大(多模式的编辑器)。
我们为什么要学习vim?
有的Liunx是自带vim,有的则不是自带的,如果没有自带,需要安装一下。
yum install -y vim
vim有很多中模式,我们现在学习三种模式:底行模式,命令模式,插入模式。
yy: 复制当前行,nyy复制n行
p: 粘贴再当前行的后面,np粘贴n次剪贴板的内容
dd: 剪切(删除)当前行,ndd操作n行
u: 撤销
ctrl + r: 重做
shift + g: 光标快速定位到文本末尾
gg: 光标快速移动到文本头
n + shift + g: 光标定位到文本的第n行
shift + 4: 光标定位到该行末尾
shift + 6: 光标定位到该行开头
w,b: 以单词为单位进行移动光标
h,j,k,l: 左、下、上、右
shift + `: 大小写快速切换
r: 替换光标所在处的字符,支持nr
shift + r: 批量化替换
x: 删除光标所在处的字符,nx删除n个
其他模式切换至命令模式,直接无脑Esc。
在底行模式的一些操作如下:
:w 只保存
:q 不保存退出
:wq 保存并退出
:reg 打开vim的寄存器面板
:syntax on 开启语法高亮
:set nu 显示行号
:set nonu 取消行号显示
:set tabstop=4 设置tab的缩进,默认为8
:set softtabstop=4 softtabstop是“逢8空格进1制表符”,前提是你tabstop=8
:set shiftwidth=4 设置程序自动缩进所使用的空格长度
:set autoindent 自动对齐上一行(这个选项会导致复制的时候代码排版混乱,可以考虑关闭,或者开启粘贴模式)
:set paste 开启粘贴模式
:set mouse=a 设置鼠标模式,默认是a
/+要搜索的内容 指定搜索
在我们之前学习中,我们讲到过如何从一个源文件形成一个可执行文件的宏观过程。今天我们用gcc每一步执行生成的文件看一看。
复习编译 + 链接:传送门
格式 gcc [选项] 要编译的文件 [选项] [目标文件]
我们将test.i打开看一下:
可见:行数如此之多,很明显头文件被展开了,还有去掉注释,宏替换。
编译:
我们将test.s打开看一下:
汇编:
我们将test.o打开看一下:
将汇编语言翻译成为可重定位二进制文件。
此时就是我们不认识的二进制的机器语言了,这里显示的是乱码。
此时文件就可以执行了吗,很显然并不能!
就算我们给test.o文件加上可执行权限,也不能执行。
因为还有一步,那就是链接!我们头文件中只是各种库函数的声明,我们还要链接其对应的库。
光有头文件是不能让代码跑起来,如果要形成可执行程序,要有头文件所对应的库。
链接:
此时生成的才是执行文件:
如何查看头文件和库?
头文件:
ls /usr/include/
库:
ls /lib64/libc*
在上述生成可执行程序的过程中,在生成test.o文件的时候,我们尝试编译了一下,并没有成功,是什么原因呢?
链接的过程是为了让我们的代码和库的代码产生关联。
test.o链接的就是下面的这个库:
库分为动态库和静态库,我们先来感性的认识一下:
动态库的文件后缀: Linux(. so) windows(. dll)
自己的程序中没有实现,通过链接别人写好的库,调用别的库中的实现。
动态链接:
动态链接本质是,自己编自己的程序,只需要在链接的时候,将需要的方法和这个方法在库中的位置,通过地址的方式关联起来,这就叫动态链接。
优缺点:
系统中很多程序都会使用C库,如果把Linux下的C库删了,那么Linux中所有的命令就跑不起来了。
静态库文件的后缀:Linux (. a) windows(. lib)
自己的程序中是将库中的相关代码,直接拷贝到自己的可执行程序中!
静态链接:
将库中所想要的方法拷贝到自己可执行的程序中,这个过程就叫做静态链接。
优缺点:
gcc中如何体现这两者的不同呢?
不同点就是他们最终形成的可执行文件体积不同。
默认一般而言,都没有自带静态库,需要我们自己安装。
安装C语言静态库:
yum install -y glibc-static
安装C+ +静态库:
yum install -y libstdc++-static
ldd + 文件名可以查看程序依赖的库和程序的链接状态:
补充:
-std=c99
c99
才支持的总结:
头文件里包的是函数的声明,C语言代码必须依赖头文件和库的,因为用的头文件和各种函数全部都需要在头文件中声明,在库文件中实现,然后在编译的时候,再把你的目标文件和库文件关联起来,这时力能形成可执行程序,否则一切都是扯淡。配环境就是在装头文件,装库。
用gdb调试的时候非常麻烦比vs麻烦得多,但是在一些特定的场景下就是需要用gdb调试。
默认生成的可执行程序是无法调试的!因为在linux
里面发布的可执行程序默认是release
版本的,并不像vs2019
那样发布的默认就是debug
版本。
这两个可执行程序运行的结果是一样的,多出来的就是调试的信息!
一段的方式读取可执行(debug):
gdb的操作:
b 行号: 打断点
info b: 查看断点
d 断点编号: 取消断点
l 行号: 显示代码
l main:显示包含main的那一行
r: run,开始运行程序,跳到第一个断点
s: step,逐语句,对应vs的F11(进入函数)
n: next,逐过程,对应vs的F10
c: continue,跳转道下一个断点
p: 查看变量
display / undisplay: 常显示 或 取消常显示
until 行号: 跳转到指定行
finish: 执行完一个函数后停下
bt: 查看函数调用堆栈
gdb知道、会用就可以了。
经过上述的学习之后,我们知道在Linux
下要是生成一个可执行文件是很麻烦的,当多个源文件一起编译的时候,一个一个生成目标文件最后再链接,麻烦死了~
项目结构:
我们在vs中生成可执行文件只需要一键点击,清理解决方案也是一键点击,非常方便。
多个源文件生成可执行,vs当中只需要点击几个按钮就可以自动生成,文件之间的关系,先编译哪个后编译哪个,哪几个文件生成可执行,一次生成几个可执行文件, vs中从来不关心。
Linux下源文件之间的关系需要我们手动维护的,说白了就是要用gcc命令。
Linux项目自动化构建工具 - make / Makefile:
make和Makefile类似于:vs当中生成解决方案。
我们在Makefile文件中通过,依赖关系和依赖方法,达到我们最终的目的,生成可执行程序。
makefile表明的是依赖关系和依赖方法。
对应的makefile如下:
依赖关系:要生成process,就需要process.c
依赖方法:如何使用process.c,形成process
此时我们只要make一下就能生成可执行程序,make clean就会将可执行程序给删除了。
注意:
目标文件和伪目标:都是目标文件最终目的都是要根据依赖关系执行依赖方法。
什么叫总是被执行:
既然有总是被执行,那么就有总不被执行:
当我们make过之后,再make就不会重复生成可执行程序,这就是总不被执行。
那么问题来了,makefile是如何识别我的exe/bin是新的还是旧的呢?
一般而言, Linux下的文件会有三种时间:
修改内容可能会引起Chang时间的变化,因为修改了文件内容文件的属性也会跟着变化,例如文件的大小。
Liunx内核对Access做了优化时间不变累计一定次数之后才会发生变化。
可执行程序和源文件对比Modify的时间来决定要不要生成新的可执行程序:
此时就更新了,源文件更新,就编译这个源文件,如果可执行程序更新,就不编译。
而总是被执行则是忽略时间对比:
不太建议将形成的目标文,件设置成伪目标而是将清理设成为目标。
见一见:
平时接触的的换行一般指的是回车 + 换行,就是另起一行。但是实际上回车和换行还是有区别的:
回车是回车,换行是换行~
在我们之前学习的C语言中,\n就是回车 + 换行,而 \r 是回车。
缓冲区刷新:
fflush()
函数来刷新一下。我们通过man来看一下:
有了上述知识,那么我们就可以利用这个特性实现一个简单的倒计时:
#include
#include
int main()
{
int cnt = 9;
while(cnt)
{
printf("%d\r", cnt);
cnt--;
fflush(stdout);
sleep(1);
}
return 0;
}
效果图:
#include
#include
#include
#define CNT 101
#define STYLE '#'
void process()
{
char arr[CNT];
memset(arr, '\0', sizeof(arr));
const char* str = "/-|\\";
int i = 0;
for(i = 0; i < 100; i++)
{
arr[i] = STYLE;
printf("\033[46;34m[%-100s]\033[0m [%d%%] %c\r", arr, i + 1, str[i % 4]);
fflush(stdout);
usleep(40000);
}
printf("\n");
}
int main()
{
process();
return 0;
}
效果图: