【Liunx】Liunx之Ubuntu入门篇
系列文章目录
留空
文章目录
- 系列文章目录
- 前言
- 什么是Liunx?
- 一、Ubuntu终端操作与Shell命令
- 二、Ubuntu文件系统结构
- 三、Ubuntu下的磁盘管理
- 四、Ubuntu下压缩与解压缩
- 五、Ubuntu用户与用户组
- 六、Ubuntu文件权限管理
- 七、Linux连接文件
- 八、vim编辑器
- 九、Linux C编程
- 十、make工具和Makefile的引入
- 十一、Makefile基本语法
- 十二、shell脚本入门
- 十三、shell脚本条件判断、函数和循环
- 总结
前言
自用
参考资料、视频:正点原子【第一期】手把手教你学Linux之Ubuntu入门篇
什么是Liunx?
Linux 是一种开源的、免费的操作系统,它的核心(Kernel)由 Linus Torvalds 于 1991 年首次发布。Linux 拥有高稳定性和安全性,被广泛应用于服务器、嵌入式设备和个人电脑。
简单来说,Linux 是一个类似 Windows 和 macOS 的操作系统,但更灵活,适合开发者和服务器环境。
Linux 和其他操作系统有什么区别?
特点 | Linux | Windows | macOS |
---|---|---|---|
价格 | 免费,人人可用。 | 需要花钱买授权(不是免费的)。 | 买苹果电脑时自带系统。 |
自由性 | 可以随意修改和定制。 | 系统固定,不能修改。 | 系统固定,不能修改。 |
安全性 | 病毒少,很安全。 | 病毒多,需要杀毒软件保护。 | 安全性高,病毒很少。 |
谁在用 | 程序员、技术人员、服务器管理员。 | 普通用户、企业、游戏玩家。 | 设计师、视频编辑、苹果用户。 |
运行速度 | 很快,对旧电脑也友好。 | 速度一般,需要好配置的电脑。 | 优化好,速度快,但只适配苹果。 |
软件支持 | 免费工具多,商用软件和游戏较少。 | 商业软件和游戏最多。 | 设计和创意类软件丰富。 |
一、Ubuntu终端操作与Shell命令
以下是对Linux命令及其常用选项:
分类 | 命令 | 功能描述 | 扩展命令(-x) |
---|---|---|---|
文件和目录操作 | ls | 显示目录中的文件和文件夹 | ls -a 显示所有文件(包括隐藏文件) |
ls -l 显示详细列表 | |||
ls -h 以可读格式显示文件大小 | |||
ls -R 递归显示子目录 | |||
ls -t 按修改时间排序 | |||
ls -S 按文件大小排序 | |||
ls -i 显示 inode 号 | |||
cd | 切换当前目录 | cd .. 切换到上一级目录 | |
cd ../.. 切换到上上一级目录,以此类推 | |||
pwd | 显示当前工作目录路径 | 无 | |
cp | 拷贝文件或目录 | cp -r 递归拷贝目录 | |
cp -i 提示是否覆盖 | |||
cp -u 只拷贝较新的文件 | |||
cp -v 显示详细输出 | |||
cp -a 保持符号链接、文件权限等不变 | |||
mv | 移动文件或目录 | mv -i 提示是否覆盖 | |
mv -u 仅在源文件较新时移动 | |||
mv -v 显示详细输出 | |||
mkdir | 创建新目录 | mkdir -p 创建父目录 | |
mkdir -v 显示详细输出 | |||
touch | 创建空文件或更新文件的访问和修改时间 | touch -c 如果文件不存在,不创建文件 | |
touch -t 设置文件时间戳 | |||
rm | 删除文件或目录 | rm -r 递归删除目录 | |
rm -f 强制删除文件 | |||
rm -i 删除前确认 | |||
rm -v 显示详细输出 | |||
rmdir | 删除空目录 | rmdir -p 删除空目录及其父目录 | |
系统信息 | uname | 显示系统信息 | uname -a 显示所有系统信息 |
uname -r 显示内核版本 | |||
uname -s 显示内核名称 | |||
uname -m 显示机器硬件名称 | |||
清理屏幕 | clear | 清除终端屏幕内容 | 无 |
文件内容操作 | cat | 显示文件内容 | cat -n 显示行号 |
cat -b 仅对非空行显示行号 | |||
cat -E 显示行尾的 $ 符号 | |||
cat -T 显示制表符为 ^I | |||
gedit | 使用文本编辑器打开文件 | gedit 文件名 如果文件存在,会打开该文件;如果文件不存在,会创建一个空文件。 | |
用户和权限管理 | sudo | 以管理员身份执行命令 | sudo -u <user> 以指定用户身份运行命令 |
sudo -i 启动交互式 shell | |||
sudo -s 启动新的 shell | |||
sudo -v 更新用户凭证 | |||
su | 切换到超级用户或其他用户身份 | su - 切换到目标用户的登录 shell | |
su <user> 切换到指定用户身份 | |||
网络和进程管理 | ifconfig | 显示网络配置信息 | ifconfig -a 显示所有接口信息 |
ifconfig <interface> 查看特定接口 | |||
ifconfig <interface> up 启用网络接口 | |||
ifconfig <interface> down 禁用网络接口 | |||
ps | 显示当前系统进程 | ps -ef 显示所有进程 | |
ps -aux 显示所有进程,包括其他用户进程 | |||
ps -u <user> 显示指定用户的进程 | |||
ps -o 自定义输出格式 | |||
top | 显示进程的实时状态 | top -u <user> 显示指定用户的进程 | |
top -d <seconds> 设置更新间隔 | |||
top -n <number> 设置显示的迭代次数 | |||
重启与关机 | reboot | 重启计算机 | 无 |
poweroff | 关机 | 无 | |
帮助与手册 | man | 显示命令的帮助文档 | man -k <keyword> 搜索命令帮助文档 |
man -f <command> 获取命令简短描述 | |||
man -a 显示所有相关手册页 | |||
文件查找 | find | 查找指定路径下的文件或目录 | find . -name "*.txt" 查找当前目录下所有 txt 文件 |
find . -type d 查找目录 | |||
find . -type f 查找文件 | |||
find . -size +100M 查找大于 100MB 的文件 | |||
find . -exec <command> {} \; 对每个找到的文件执行命令 | |||
文件类型查看 | file | 查看文件类型 | file -i 查看文件的 MIME 类型 |
file -b 仅显示文件类型,不显示文件名 | |||
磁盘管理 | du | 查看文件或目录的磁盘使用情况 | du -h 以可读格式显示大小 |
du -s 显示总计 | |||
du -a 显示所有文件 | |||
du -c 显示总计 | |||
du -d <level> 指定递归深度 | |||
df | 查看磁盘空间使用情况 | df -h 以可读格式显示磁盘空间 | |
df -T 显示文件系统类型 | |||
df -i 显示 inode 使用情况 | |||
sync | 将数据同步写入磁盘 | 无 |
后续用到命令直接百度!
二、Ubuntu文件系统结构
在 Linux 系统中,/
(斜杠)是根目录,它是整个文件系统的起点和最顶层目录。可以把根目录理解为文件系统的“根基”,所有其他目录和文件都是从根目录开始分支和组织的。
整个Linux系统有且只有一棵从根目录开始的目录树,如下图所示。
下面是 Linux 文件系统目录结构的内容:
目录 | 内容描述 |
---|---|
/bin | 存放基本系统命令,如 cat 、cp 、mkdir 等。用于系统启动和修复。 |
/boot | 存放开机时需要的文件,如启动管理程序 grub2 。 |
/dev | 存放设备文件,用于与硬件设备(如声卡、硬盘、光驱等)进行交互。 |
/etc | 系统配置文件的存放目录,包含网络配置、用户配置等。 |
/home | 用户的家目录,存放普通用户的个人文件。 |
/lib | 存放系统命令(bin 和 sbin 目录下的命令)所需的共享库文件。 |
/lib32 //lib64 | 存放 32 位和 64 位系统所需的二进制共享库文件。 |
/lost+found | 用于存放系统崩溃或关机时丢失的文件碎片,系统修复时会检查并恢复这些文件。 |
/media | 用于挂载可移动设备,如 CD、DVD、U盘等。 |
/mnt | 用于临时挂载存储设备,通常由管理员手动挂载设备。 |
/opt | 存放第三方软件安装目录。 |
/proc | 存放进程信息和内核信息,提供系统运行时的数据,不占用硬盘空间。 |
/root | root 用户的家目录。 |
/run | 存储系统启动以来的信息,系统重启后会清空。 |
/sbin | 存放系统管理员(root)使用的命令,如磁盘管理、网络管理等。 |
/srv | 存放服务相关的数据文件,如 Web 服务、FTP 服务的数据。 |
/sys | 存放与系统硬件相关的信息,类似于 proc ,用于内核与硬件之间的通信。 |
/tmp | 存放临时文件,程序运行时产生的临时数据,系统重启后通常会清空。 |
/usr | 存放系统程序和库文件,类似于 Windows 系统中的 Program Files 目录。 |
/var | 存放内容变化频繁的文件,如日志文件、缓存文件、邮件等。 |
绝对路径和相对路径
特性 | 绝对路径 | 相对路径 |
---|---|---|
起始点 | 从根目录 / 开始,表示完整路径。 | 从当前工作目录开始,表示相对于当前位置的路径。 |
访问方式 | 无论当前目录在哪里,路径始终指向同一个目标。 | 路径取决于当前所在的目录。 |
表示方式 | 必须以 / 开头,包含了从根目录开始的所有目录。 | 不以 / 开头,只描述相对位置。 |
示例 | /home/user/documents/file.txt | documents/file.txt (当前工作目录为 /home/user ) |
具体例子:假设你有一个文件 report.txt
存放在 /home/user/docs/
目录下。
/
├── home/
│ └── user/
│ ├── docs/
│ │ └── report.txt
│ └── images/
│ └── photo.jpg
└── var/
└── log/
└── system.log
- 绝对路径访问:不管当前在哪个目录,你想访问
report.txt
文件。
/home/user/docs/report.txt
无论你当前在 /home/user/images
目录还是 /var/log
目录,使用这个路径始终能够找到 report.txt
文件,因为绝对路径从根目录开始,始终指向系统中同一个文件。
- 相对路径访问:可以使用
.
(当前目录)和..
(上一级目录)来导航。
(1)若从 /home/user
目录访问 report.txt
文件
./docs/report.txt 或者 docs/report.txt
从当前目录( /home/user
)进入 docs
子目录,然后访问 report.txt
文件。
(2)若从 /home/user/images
目录访问 report.txt
文件
../docs/report.txt
..
表示上一级目录(即 /home/user
),然后进入 docs
子目录,找到 report.txt
文件。
三、Ubuntu下的磁盘管理
【待更改】
1. Linux 磁盘管理基本概念
在 Linux 中,磁盘管理和 Windows 有很大的区别。Windows 使用的是 “分区” 的概念,而 Linux 使用的是 “挂载点”。
1.1 挂载点与磁盘设备文件
- 挂载点:就是将硬盘分区挂载到某个目录,使得用户可以像访问文件夹一样访问磁盘内容。
- 设备文件:每个磁盘和分区在
/dev/
目录下都有一个设备文件,如/dev/sda1
、/dev/sdb1
,这些文件表示硬盘和分区。
1.2 /etc/fstab 文件
/etc/fstab
是系统文件,记录了磁盘分区及挂载点的配置。它描述了哪些设备在启动时会自动挂载到指定目录。
2. 查看磁盘与分区
- 使用命令
ls /dev/sd*
来查看当前系统中所有的磁盘设备。
示例:
/dev/sda
:第一个硬盘/dev/sdb
:第二个硬盘(可能是 U 盘等外部设备)/dev/sdb1
:第二个硬盘的第一个分区
3. 常用磁盘管理命令
命令 | 功能 | 说明 |
---|---|---|
fdisk | 磁盘分区 | 用于对磁盘进行分区管理,fdisk /dev/sdb 会让你对 /dev/sdb 进行分区操作。 |
mkfs | 格式化磁盘分区 | 格式化分区并创建文件系统,例如 mkfs -t vfat /dev/sdb1 格式化为 FAT 文件系统。 |
mount | 挂载磁盘 | 挂载磁盘或分区到目录,例如 mount /dev/sdb1 /mnt/tmp 将 /dev/sdb1 挂载到 /mnt/tmp 。 |
umount | 卸载磁盘 | 卸载已挂载的磁盘或分区,例如 umount /mnt/tmp 卸载挂载点 /mnt/tmp 。 |
4. 磁盘分区命令 fdisk
fdisk
用于管理磁盘分区,可以创建、删除、修改分区。使用时需要指定磁盘设备,如 /dev/sdb
。
常用选项与子命令
子命令 | 功能 |
---|---|
p | 显示当前分区信息 |
n | 创建新分区 |
t | 更改分区类型 |
d | 删除现有分区 |
a | 设置分区为启动分区 |
w | 保存并写入分区表 |
q | 不保存退出 |
分区操作示例
-
启动
fdisk
分区工具:sudo fdisk /dev/sdb
-
输入
m
查看帮助,选择对应的操作:- 输入
n
创建新分区 - 输入
w
保存更改
- 输入
5. 格式化分区命令 mkfs
创建文件系统的命令。常用于格式化硬盘分区。
常用选项
选项 | 功能 |
---|---|
-t | 指定文件系统类型(如 vfat , ext4 等) |
-v | 显示详细执行过程 |
格式化示例
-
格式化为
vfat
文件系统:sudo mkfs -t vfat /dev/sdb1
-
格式化为
ext4
文件系统:sudo mkfs -t ext4 /dev/sdb1
6. 挂载分区命令 mount
将硬盘分区挂载到指定目录,之后就可以通过该目录访问磁盘内容。
常用选项
选项 | 功能 |
---|---|
-t | 指定文件系统类型(如 vfat , ext4 等) |
-o ro | 只读模式挂载 |
-o rw | 读写模式挂载 |
-v | 显示详细的执行过程 |
挂载示例
-
创建挂载点目录:
sudo mkdir /mnt/tmp
-
挂载分区
/dev/sdb1
到/mnt/tmp
:sudo mount -t vfat /dev/sdb1 /mnt/tmp
7. 卸载分区命令 umount
当磁盘或分区不再需要时,使用 umount
命令将其从挂载点卸载。
常用选项
选项 | 功能 |
---|---|
-v | 显示详细信息 |
-r | 如果无法卸载,尝试以只读方式卸载 |
-a | 卸载 /etc/mtab 中的所有文件系统 |
卸载示例
- 卸载挂载点
/mnt/tmp
:sudo umount /mnt/tmp
8. 磁盘管理操作流程总结
以下是磁盘管理的常见操作流程,帮助你理解如何操作磁盘和分区。
操作 | 命令 | 说明 |
---|---|---|
查看磁盘设备 | ls /dev/sd* | 查看所有磁盘设备文件,如 /dev/sda , /dev/sdb 等。 |
分区操作 | sudo fdisk /dev/sdb | 对 /dev/sdb 磁盘进行分区。常用子命令有 n 创建新分区。 |
格式化磁盘分区 | sudo mkfs -t vfat /dev/sdb1 | 格式化 /dev/sdb1 为 vfat 文件系统。 |
挂载磁盘分区 | sudo mount -t vfat /dev/sdb1 /mnt/tmp | 将 /dev/sdb1 挂载到 /mnt/tmp 目录。 |
卸载磁盘分区 | sudo umount /mnt/tmp | 卸载 /mnt/tmp 目录下的挂载分区。 |
四、Ubuntu下压缩与解压缩
【待更改】
Linux 常用压缩和解压缩工具
工具 | 压缩命令 | 解压命令 | 描述 |
---|---|---|---|
gzip | gzip xxx | gzip -d xxx.gz | 压缩和解压 .gz 格式文件 |
gzip -r xxx | gzip -rd xxx.gz | 压缩/解压文件夹(递归) | |
bzip2 | bzip2 -z xxx | bzip2 -d xxx.bz2 | 压缩和解压 .bz2 格式文件 |
tar | tar -vcf test.tar test | tar -vxf test.tar | 打包和解包文件(不含压缩) |
tar -vxjf xxx.tar.bz2 | tar -vcjf xxx.tar.bz2 xxx | 压缩/解压 .tar.bz2 格式文件 | |
tar -vxzf xxx.tar.gz | tar -vczf xxx.tar.gz xxx | 压缩/解压 .tar.gz 格式文件 | |
rar | rar a xxx.rar xxx | rar x xxx.rar | 压缩和解压 .rar 格式文件 |
zip | zip -rv xxx.zip xxx | unzip -v xxx.zip | 压缩和解压 .zip 格式文件 |
五、Ubuntu用户与用户组
1. Linux 用户
在 Linux 系统中,用户是操作系统中非常重要的概念。Linux 是一个多用户操作系统,每个用户都有自己独立的权限。Ubuntu 和其他 Linux 系统通常包含三种用户:
用户类型 | 描述 | 权限 | 常用命令 |
---|---|---|---|
初次创建的用户 | 安装系统时创建的第一个用户,拥有部分管理员权限,可以进行一般的系统管理操作。 | 拥有比普通用户更多的权限,但不如 root 强大。 | adduser username 创建新用户 |
root用户 | 系统的超级管理员,具有最高权限,可以执行任何操作。 | 完全控制系统,安装软件、修改设置、删除文件 等。 | sudo su 切换到 root 用户 |
普通用户 | 普通用户,权限较低,通常只能操作自己的文件和目录。 | 只能访问自己的文件和目录,无法执行系统管理任务。 | sudo 提升权限,如 sudo command |
用户的基本信息存储:
项 | 描述 | 相关命令 |
---|---|---|
用户信息存储 | 用户信息存储在 /etc/passwd 文件中,包含用户名、UID、家目录等信息 | cat /etc/passwd 查看用户信息 |
用户密码存储 | 用户的密码存储在 /etc/shadow 文件中,存储的是加密后的密码 | sudo cat /etc/shadow (仅 root 可查看) |
UID (用户 ID) | 用户唯一标识符,root 用户的 UID 为 0,普通用户的 UID 通常从 1000 开始 |
2. Linux 用户组
概念 | 描述 | 相关命令 |
---|---|---|
用户组 | 将多个用户归为一组,方便统一管理权限 | |
用户组 ID (GID) | 每个用户组有唯一的标识符,用 GID 区分不同的组 | |
用户组信息存储 | 用户组信息存储在 /etc/group 文件中,包含组名、GID 和组成员信息 | cat /etc/group 查看所有用户组信息 |
用户与用户组关系 | 用户可以属于多个组,文件的访问权限根据用户所属的组来管理 | groups username 查看用户所属组 |
用户组与权限 | 文件的访问权限可以基于用户组进行设置,组内用户共享文件访问权限 | chown 和 chmod 设置文件所有权和权限 |
3. 创建用户与用户组
操作 | 命令 | 说明 |
---|---|---|
创建用户 | sudo adduser username | 创建新用户,并自动为其创建家目录和设置密码 |
查询用户信息 | finger username | 查询指定用户的信息 |
修改用户密码 | sudo passwd username | 修改用户的密码 |
删除用户 | sudo deluser username | 删除指定用户 |
删除用户及家目录 | sudo deluser --remove-home username | 删除用户及其家目录 |
创建用户组 | sudo addgroup groupname | 创建新用户组 |
查询用户组信息 | groups username | 查询指定用户所属的组 |
删除用户组 | sudo delgroup groupname | 删除指定用户组 |
添加用户到组 | sudo usermod -aG groupname username | 将指定用户添加到某个用户组中 |
删除用户组成员 | sudo deluser username groupname | 将用户从某个用户组中删除 |
4. 用户组与文件权限
概念 | 描述 | 相关命令 |
---|---|---|
文件权限 | 每个文件有 3 类权限:用户权限、组权限、其他用户权限 | |
权限字符 | r (读权限),w (写权限),x (执行权限),- 表示无权限 | |
文件所有者与组 | 每个文件都有一个所有者(用户)和一个所属用户组,决定访问权限 | ls -l 显示文件权限 |
修改文件权限 | chmod 命令用于修改文件的权限 | chmod 755 filename 设置文件权限 |
修改文件所有者 | chown 命令用于修改文件的所有者和所属用户组 | sudo chown user:group filename 修改文件所有者和组权限 |
组权限的使用 | 用户组可以共享文件访问权限,管理员通过修改文件的组权限来控制访问 | chmod 770 filename 设置文件的组权限 |
5. 常见命令汇总
命令 | 功能 |
---|---|
adduser | 创建新用户 |
deluser | 删除用户 |
addgroup | 创建新用户组 |
delgroup | 删除用户组 |
usermod | 修改用户属性,如添加用户到组 |
passwd | 修改用户密码 |
groups | 查看用户所属的组 |
finger | 查看用户的详细信息 |
cat /etc/passwd | 查看系统中所有用户的信息 |
cat /etc/group | 查看系统中所有用户组的信息 |
chown | 修改文件或目录的所有者和所属组 |
chmod | 修改文件或目录的权限 |
id | 显示当前用户的 UID 和 GID 信息 |
ls -l | 显示文件的详细权限信息 |
6. UID 和 GID
概念 | 描述 | 默认值 |
---|---|---|
UID (用户 ID) | 用户唯一标识符,root 用户的 UID 为 0,普通用户的 UID 从 1000 开始 | root 用户 UID = 0 |
GID (组 ID) | 用户组唯一标识符,用于区分不同的用户组 | root 组 GID = 0 |
六、Ubuntu文件权限管理
1. 文件权限概述
Linux 文件权限用来限制和管理用户对文件的操作权限。权限分为三种:
- r (read): 读取权限,允许查看文件内容。
- w (write): 写入权限,允许修改文件内容。
- x (execute): 执行权限,允许将文件作为程序执行。
也可以用权限以符号和二进制数值表示:
r = 4
,w = 2
,x = 1
,无权限为0
。
2. 文件权限格式
文件权限的格式如下:
-xxxxxxxxx(例:-rw-rw-r--)
- 第一位:文件类型。
-
表示普通文件。 - 接下来,每三个为一组,表示不同用户的权限:
- 第一组:文件拥有者的权限。
- 第二组:组内用户的权限。
- 第三组:其他用户的权限。
3. 具体例子
第一步,我们先在~
下创建一个文件a.c
touch a.c
第二步。查看a.c
的详细信息
ls a.c -l
输出以下信息
-rw-rw-r-- 1 lty lty 0 12月 10 12:04 a.c
-
:普通文件。rw-rw-r--
:-
- 拥有者
rw-
:读、写权限。 - 组用户
rw-
:读、写权限。 - 其他用户
r--
:只读权限。
- 拥有者
1
:硬链接数。lty lty
:拥有者和用户组。0
:文件大小为 0 字节。12月 10 12:04
:最后修改时间。a.c
:文件名。
4. 修改文件权限
4.1 使用 chmod
命令
chmod
用于修改文件或目录的权限。
chmod 更改权限 文件/目录
方式一:符号法
使用符号修改权限:
u
:文件拥有者 (user)。g
:所属组 (group)。o
:其他用户 (others)。a
:所有用户 (all。- 权限操作符:
+
添加权限,-
移除权限,=
设置权限。
示例
chmod u+x a.c # 给文件拥有者添加执行权限
chmod g-w a.c # 移除组用户的写权限
chmod o=r a.c # 设置其他用户仅有读取权限
具体例子:
【输入】ls a.c -l #查看a.c文件信息
【输出】rw-rw-r-- 1 lty lty 0 12月 10 12:04 a.c #a.c的详细信息
【输入】./a.c #执行a.c文件
【输出】bash: ./a.c: 权限不够 #无执行权限
【输入】chmod u+x a.c #更改权限,添加拥有者的执行权
【输入】ls a.c -l #再次查看a.c文件信息
【输出】-rwxrw-r-- 1 lty lty 0 12月 10 12:04 a.c #显示拥有者已有执行权
【输入】./a.c #再次执行(无显示,就是可执行)
方式二:数字法
使用权限值修改权限,每组三位,分别表示 u
, g
, o
的权限:
- 权限值为
rwx
的加和,r = 4
,w = 2
,x = 1
,例如:rw- = 6
,r-- = 4
。
示例
chmod 764 a.c # 设置拥有者权限为 rwx,组用户为 rw-,其他用户为 r--
具体例子:
递归修改
使用 -R
递归修改目录及子目录下的所有文件权限:
chmod -R 755 /path/to/directory
4.2 使用 chown
命令修改文件所属用户
用法
chown [用户:组] 文件/目录
示例
chown root a.c # 修改文件拥有者为root
chown root:root a.c # 修改文件拥有者为root,组为root
具体例子:
【更改前】-rwxrw-r-- 1 lty lty 0 12月 10 12:04 a.c
【更改后】-rwxrw-r-- 1 root lty 0 12月 10 12:04 a.c
文件拥有者从lty
变为root
接下来,把用户组也更改看看
组也从从lty
变为root
一个一个改太麻烦了,能不能同时改呢?可以的!
sudo chown lty.lty a.c (chown 使用者.使用组 文件)
递归修改
chown -R user:group /path/to/directory
七、Linux连接文件
Linux 提供了两种文件连接方式:硬连接和软连接(符号连接)。
1. 硬连接(Hard Link)
硬连接就像你给同一本书起了两个不同的名字。无论你改了哪个名字,书的内容都不变,删掉一个名字,另一个名字还是能继续访问这本书。
举例: 你有一个文件 file1.txt
,然后创建一个硬连接 file2.txt
:
ln file1.txt file2.txt
file1.txt
和file2.txt
其实是同一个文件,只是不同的名字。删除其中一个,另一个依然有效,文件内容不受影响。- 如果
file1.txt
有10kb,创建了一个硬连接file2.txt
,那么file1.txt
和file2.txt
都会占用相同的存储空间(10KB)。
2. 软连接(Symbolic Link)
软连接就像是文件的快捷方式,它是一个指向原文件的链接。你可以把它看作是一个快捷方式,指向真实的文件。如果原文件被删除,快捷方式就会失效,打不开了。
举例: 你有一个文件 file1.txt
,然后创建一个软连接 link_to_file.txt
:
ln -s file1.txt link_to_file.txt
link_to_file.txt
就是file1.txt
的一个快捷方式。如果你删除了file1.txt
,link_to_file.txt
就打不开了,它会失效。- 如果
file1.txt
有10kb,创建了一个软连接link_to_file.txt
,那么link_to_file.txt
只占用存储路径的空间,比如路径file1.txt
大概是 12 字节。所以,软连接文件的大小通常很小。
3. 硬连接 vs 符号连接
特性 | 硬连接(Hard Link) | 软连接(Symbolic Link) |
---|---|---|
inode | 相同,硬连接和源文件共享同一个inode | 不同,软连接有自己的独立 inode |
依赖源文件 | 源文件被删除,硬连接正常使用 | 源文件被删除,软连接失效 |
跨文件系统 | 不支持,硬连接只能在同一文件系统内创建 | 支持,可以跨文件系统创建软连接 |
连接目录 | 不支持,不能为目录创建硬连接 | 支持,可以连接到目录 |
创建方式 | ln 源文件 硬连接文件 | ln -s 源文件 软连接文件 |
数据一致性 | 修改源文件或硬连接,内容同步变化 | 修改源文件,内容同步变化,但软连接仍指向路径 |
灵活性 | 结构简单,适用于保护重要文件,防止误删除 | 灵活性高,适用于创建快捷方式 |
路径依赖性 | 不存在,所有硬连接实际上都是同一个文件 | 依赖源文件路径,建议使用绝对路径以避免连接失效 |
符号表示 | 无特别的显示方式,在 ls -l 中与普通文件相同 | ls -l 中显示软连接,以 -> 指向源文件 |
4. 具体例子
首先,创建一个文件hello.c
touch hello.c
在文件里添加以下内容
#include <stdio.h>
int main(void)
{
printf("hello world!!\n");
return 0;
}
写好后,我们尝试去执行这个文件,看看能不能输出
我们直接运行 C 源代码文件 hello.c
,显示错误,但是我们代码是正确的,为什么呢?
错误原因
- 源代码文件不能直接执行:
- 虽然我们给
hello.c
文件添加了可执行权限,但它是一个源代码文件,而不是可执行程序,不能直接执行。直接运行源代码文件会导致 Bash 尝试将其作为脚本执行,从而产生语法错误。
- 虽然我们给
- 错误信息解释:
- 错误信息
未预期的记号 "(" 附近有语法错误
表示 Bash 试图解析 C 语言的语法,但它并不理解 C 语言的语法结构。
- 错误信息
正确的工作流程
正确运行 C 代码,要遵循以下步骤:
-
编译源代码:
- 使用
gcc
编译源代码文件hello.c
,生成可执行文件。
gcc 需要编译的文件 -o 生成的可执行文件
- 使用
-
运行可执行文件:
- 运行生成的可执行文件
hello
,而不是源代码文件hello.c
:
./可执行的文件
- 运行生成的可执行文件
gcc hello.c -o hello
./hello
OK,终于成功运行了!
回归正题,将hello
文件分别进行硬连接和软连接。
4.1 硬连接
创建硬连接的命令:
ln [源文件] [硬连接文件]
我们创建两个硬连接hello1
和 hello2
ln hello hello1
ln hello hello2
(1)硬连接和源文件共享同一个inode,什么是inode,怎么看呢?
简单来说,inode 就是文件的“身份证”,它记录了文件的基本信息,帮助操作系统管理文件。我们输出以下命令,用来列出所有以 hello
开头的文件,并显示它们的 inode 号码。
ll -i hello*
ll
:是ls -l
命令的简写,用于列出当前目录中所有文件和子目录的详细信息;-i
:用于显示文件的 inode号码;hello\*
:这是一个通配符,用于匹配所有以hello
开头的文件或目录。例如,hello1
、hello2
、hello123
等都符合这个模式。
从输出可知,hello
、hello1
和 hello2
这三个文件都指向相同的 inode(3028616
),并且它们的链接数都变成了 3。这意味着我们已经创建了三个硬连接指向同一个文件。
(2)我们删掉一个硬连接,或者直接删掉源文件,看看其他的还能打开吗?
删掉其他的硬连接和源文件,其他的硬连接都能这些执行!只有所有硬连接删除后,文件内容才会被删除。
(3)那么如果我们改动一个硬连接的内容,其他的会跟着改变吗?
我们将hello.c
创建一个硬连接hello1.c
,和源文件内容一致
把硬连接hello1.c
改动一下,点击保存
再打开hello.c
修改任意硬连接都会影响源文件和所有硬连接!!!
4.2 软连接
创建硬连接的命令:
ln -s [源文件] [软连接文件]
我们创建两个硬连接helloA
和 helloB
ln -s hello helloA
ln -s hello helloB
(1)软连接(符号链接)与硬连接不同,它有自己的 inode 和内容。
但它的 inode 并不直接存储文件的数据内容,而是存储指向目标文件路径的信息。
(2)删掉一个软连接,或者删掉源文件,看看其他软连接还能正常执行吗?
删除软连接本身并不会影响源文件或其他软连接。软连接只是一个指向目标文件路径的快捷方式,它与目标文件是独立的。其他指向同一源文件的软连接仍然有效,能够正常执行。
如果你删除了源文件,软连接会失效,指向的路径变得无效。所有指向该文件的软连接将变为悬挂链接。这些软连接将无法访问源文件内容,因为源文件已经不存在了。
(3)那么如果我们改动一个软连接的内容,其他的会跟着改变吗?
我们将hello.c
创建一个软连接helloA.c
,和源文件内容一致
把软连接helloA.c
改动一下,点击保存
再打开hello.c
修改任意软连接都会影响源文件和所有软连接!!!
(4)符号连接(软连接) 推荐使用绝对路径,为什么呢?
我们先创建两个文件夹test
和test1
,将上面的hello
复制到test
中并重命名为A
。
创建A
的两个软连接,一个使用相对路径,另一个使用绝对路径。具体如下:
代码:
ln -s A B # 使用相对路径创建软连接B
ln -s /home/lty/test/A /home/lty/test/C # 使用绝对路径创建软连接C
mv B /home/lty/test1 # 把B移动到test1
mv C /home/lty/test1 # 把C移动到test1
为什么使用绝对路径创建软连接C正常执行,使用相对路径创建软连接B失效了呢??
相对路径:
-
如果,我们在
/home/lty/test
目录下创建软连接,使用相对路径:ln -s hello helloC
-
这样
helloC
会指向hello
。但如果我们将hello
移动到其他目录(比如/home/lty/test1
),它将无法找到hello
,因为它使用的是相对路径,相对路径依赖于当前位置。
绝对路径:
-
使用绝对路径创建软连接:
ln -s /home/lty/hello /home/lty/test/hello
-
不管你移动软连接
hello2
到哪里,它都会始终指向/home/lty/hello
,因为使用的是绝对路径,路径不会受到你当前目录的影响。
注:我们使用软连接更多!软连接更常用,因为它更灵活。它可以跨文件系统和分区创建,还能链接目录。软连接是路径指向,修改起来方便,而硬连接指向文件本身,不能轻易修改。删除源文件时,软连接会变成无效链接,而硬连接只要还有链接存在,文件就不会被删除。所以,软连接更适合管理文件和目录。
最后,可以使用以下命令删掉所有的上面示例的文件!
rm -rf test test1 hello*
八、vim编辑器
下载vim编辑器
sudo apt install vim
使用 vim 编辑器打开文件
vi xxx
Vim 编辑器的操作主要分为三种模式:
- 一般模式(指令模式):这是 Vim 启动后默认的模式,用户可以在此模式下进行文件浏览、光标移动、删除、复制等操作。此模式下无法直接输入文本,所有编辑命令都通过键盘输入。
- 编辑模式:从一般模式进入编辑模式后,用户可以自由编辑文本。此模式下可以插入文本、删除字符等。退出编辑模式后返回一般模式。
- 命令行模式(底行模式):用户可以在此模式下执行保存、退出、查找等命令。
1. vim编辑器模式
模式 | 命令 | 功能 |
---|---|---|
一般模式 | ESC | 退出编辑模式,进入一般模式 |
: | 进入命令行模式(底行模式) | |
/ | 进入命令行模式进行查找(向下查找文本) | |
? | 进入命令行模式进行查找(向上查找文本) | |
编辑模式 | i | 在光标前插入,进入编辑模式 |
I | 在光标所在行的行首插入,进入编辑模式 | |
a | 在光标后插入,进入编辑模式 | |
A | 在光标所在行的行尾插入,进入编辑模式 | |
o | 在光标所在行下方新建一行并进入编辑模式 | |
O | 在光标所在行上方新建一行并进入编辑模式 | |
s | 删除光标所在字符并进入编辑模式 | |
r | 替换光标所在字符 | |
ESC | 退出编辑模式,返回一般模式 | |
命令行模式 | w | 保存文件 |
q | 退出 vi/vim,若文件已修改则提示保存 | |
wq | 保存并退出 vi/vim | |
q! | 强制退出 vi/vim,不保存文件 | |
/xxx | 向下查找文本 xxx | |
?xxx | 向上查找文本 xxx |
2. 常用快捷键命令
命令 | 功能 |
---|---|
h | 光标左移一个字符 |
l | 光标右移一个字符 |
j | 光标下移一行 |
k | 光标上移一行 |
nG | 移动光标到第 n 行的行首 |
n+ | 光标下移 n 行 |
n- | 光标上移 n 行 |
Ctrl+f | 屏幕向下翻一页,相当于下一页 |
Ctrl+b | 屏幕向上翻一页,相当于上一页 |
cc | 删除整行并进入修改模式 |
dd | 删除当前行 |
ndd | 删除当前行及其下方 n 行 |
x | 删除光标所在字符 |
X | 删除光标前的字符 |
nyy | 复制当前行及其下方 n 行 |
p | 粘贴最近复制的内容 |
P | 将复制的内容粘贴到光标所在行上方 |
. | 重复上一次的操作 |
u | 撤销上一步操作 |
yy | 复制光标所在行 |
nyy | 复制光标所在行及其下方 n 行 |
九、Linux C编程
在 Ubuntu 下,C 语言编程的过程分为代码编写和编译两个部分。首先,使用 VIM 编辑器创建并编写代码。然后,使用 GCC 编译器进行编译。
1. VIM编写C程序
首先,需要配置 VIM 编辑器,设置 TAB 键为 4 字节和显示行号,以便更好地编辑 C 代码。
输入以下命令,设置VIM。
sudo vi /etc/vim/vimrc
进入界面,按a
进入编辑模式,在最末尾输入以下命令:
Set ts=4
set nu
然后,按ESC
退出编辑模式,输入:wq
退出保存,设置完成!
开始编写代码啦!!
创建一个hello.c
文件
vi hello.c
输入以下代码:
1 #include <stdio.h>
2 int main(int argc, char *argv[])
3 {
4 printf("hello,world\n");
5 return 0;
6 }
然后按ESC
退出编辑模式,输入:wq
退出保存。
2. GCC编译C程序
接下来,我们要使用 GCC 编译器编译我们刚刚写的代码。
GCC 编译器通过命令行使用,最基本的命令格式为:
gcc [选项] [编译文件名]
选项 | 说明 | 示例命令 |
---|---|---|
-c | 只编译源文件,不进行链接,生成目标文件(.o )。 | gcc -c main.c |
-o | 指定输出的可执行文件或目标文件名,默认为 a.out 。 | gcc -o myprogram main.c |
-g | 生成调试信息,便于调试工具(如 GDB)使用。 | gcc -g main.c |
-O | 进行优化,提高执行效率。 | gcc -O main.c |
-O2 | 更高的优化级别,优化效果更显著,但编译过程较慢。 | gcc -O2 main.c |
-Wall | 启用大部分的编译警告,帮助检测潜在的错误。 | gcc -Wall main.c |
-Werror | 将所有警告视为错误,编译失败时终止。 | gcc -Wall -Werror main.c |
-lm | 链接数学库,常用于数学运算函数。 | gcc main.c -lm |
那么,我们就在命令行中输入命令
gcc hello.c
输出了一个a.out
,这个就是hello.c
对应的可执行文件
正常执行!
如果想要自己命名可执行文件可以输入gcc [编译文件名] -o [自定义可执行文件名]
gcc hello.c -o haha
3. GCC编译错误和警告
GCC 编译器在编译时会检查代码中的错误,我们怎么查看错误信息呢?
首先,我们先写一个有错误
和警告
的代码,在命令行输入vi hello.c
打开
#include <stdio.h>
int main(int argc, char *argv[])
{
int a = 1;
int b = 2
printf("a + b =\n",a + b);
return 0;
}
编译:
gcc hello.c
编译失败
hello.c:7:9: error: expected ‘,’ or ‘;’ before ‘printf’
7 | printf("a + b =\n",a + b);
这个错误信息是说,错误
:在第 7 行的 printf
前,缺少一个逗号(,
)或分号(;
);第七行,printf
语句中,"a + b =\n"
这个字符串后面的 a + b
作为参数没有正确传递。
OK!我们先修改第一个看看!
重新编译
刚刚的错误
消失了,只剩下一个警告,再把这个警告修改一下!
无警告无错误。
正常执行!
4. 编译流程
GCC 编译器的编译流程是:预处理、编译、汇编和链接。
编译阶段 | 描述 |
---|---|
**预处理 ** | 展开头文件(如 #include )、替换宏(如 #define )、解析条件编译(如 #if )并修改源代码。 |
编译 | 将预处理后的代码转换为汇编语言代码。 |
汇编 | 将汇编代码转为二进制目标文件(.o 文件)。 |
链接 | 将多个目标文件(.o 文件)和库文件链接成最终的可执行文件,涉及静态库和动态库。 |
对于简单的单文件程序,GCC 会自动完成这些步骤,例如运行 gcc main.c
会直接生成可执行文件 a.out
。
在 Linux 下,如果程序只有一两个 C 文件,可以通过在终端执行 gcc
命令来编译;但如果文件数目增多(如几十、上百甚至上千个文件),使用终端命令就不再方便。为了解决这个问题,我们就可以使用 Makefile 和 make 工具。
十、make工具和Makefile的引入
make工具
make
是一个“智能”的编译助手。在编程时,我们通常会编写多个源文件,并希望将它们编译成一个可执行程序。make
能够自动找出哪些源文件已经改变,并仅重新编译这些改变的文件,而不是全部重新编译。这样,可以大大节省编译时间。
Makefile
Makefile
是一个配置文件,它告诉make
如何编译程序。在这个文件中,我们列出了要编译的源文件、编译时需要的选项、以及源文件之间的依赖关系等。当我们在命令行中运行make
命令时,make
会读取Makefile
中的信息,并按照指定的规则执行编译任务。
1. 具体例子
接下来,我们写一个多个文件的代码,试试看普通编译和使用make/makefile有什么区别!
首先,我们将构建一个简单的C程序,包含三个源文件 main.c
, input.c
, calcu.c
,和两个头文件 input.h
和 calcu.h
。这个程序的功能是从键盘输入两个整数,计算它们的和并输出结果。
/c_test1
│
├── main.c
├── input.c
├── calcu.c
├── input.h
└── calcu.h
完整代码:
main.c
#include <stdio.h>
#include "input.h"
#include "calcu.h"
int main(int agrc, char* argv[])
{
int a,b,num;
input_int(&a, &b);
num = add(a,b);
printf("%d + %d = %d",a,b,num);
}
input.c
#include <stdio.h>
#include "input.h"
void input_int(int *a, int *b)
{
printf("input two num:");
scanf("%d %d", a, b);
printf("\r\n");
}
calcu.c
#include "calcu.h"
int add(int a, int b)
{
return (a + b);
}
input.h
#ifndef _INPUT_H
#define _INPUT_H
void input_int(int *a, int *b);
#endif
calcu.h
#ifndef _CALCU_H
#define _CALCU_H
int add(int a, int b);
#endif
完成!
2. 两种方法编译
2.1 普通编译
gcc main.c calcu.c input.c -o add # 编译
./add # 执行
如果我们更改了其中一个代码
vi calcu.c
我们每次编译都需要写一串xxx.c
,而且我们只更改了一个文件,所有的文件都重新进行编译。
若我们是几十个或几百个文件,每次就都要写一串xxx.c
,虽然可以复制粘贴,但是文件多时,我们可能会忘哪些编译哪些没有编译过,容易遗忘或是出错,全部编译就特别费时间!
2.1 使用make工具编译
首先,要创建一个Makefile
文件
vi Makefile
添加以下代码:
保存退出,然后在命令行
make
可能原因:
- Makefile 中命令缩进没有使用 TAB 键!
- VI/VIM编辑器使用空格代替了 TAB 键,修改文件/etc/vim/vimrc,在文件最后加上代码:
set noexpandtab
成功编译后,会出现四行代码
gcc -c main.c # 编译 main.c 文件生成目标文件 main.o
gcc -c input.c # 编译 input.c 文件生成目标文件 input.o
gcc -c calcu.c # 编译 calcu.c 文件生成目标文件 calcu.o
gcc -o main main.o input.o calcu.o # 链接三个目标文件生成可执行文件 main
然后执行main
,正确执行!
然后,我们修改其中一个代码
vi calcu.c
make
自动更新我们修改过的代码,然后执行看看!
正常执行!而且只需要写一次Makefile
文件,以后就可以很便捷的编译代码啦。
最后,一键删除。
十一、Makefile基本语法
这里只简单的把上节Makefile内容进行优化和理解,深入的学习可参考《跟我一起写 Makefile》。
上一节,我们的Makefile
虽然比普通编译便捷,但是还不是特别便捷!
三个文件可以一个一个列出来,难道几十个几百个也要一个一个列出嘛
main: main.o input.o calcu.o
gcc -o main main.o input.o calcu.o
main.o: main.c
gcc -c main.c
input.o: input.c
gcc -c input.c
calcu.o: calcu.c
gcc -c calcu.c
clean:
rm *.o
rm main
如果学习了Makefile基本语法,上面的代码就可以简化成下面这样:
objects = main.o input.o calcu.o # 定义目标文件列表
main: $(objects) # main 目标依赖于 $(objects)(即 main.o, input.o, calcu.o)
gcc -o main $(objects) # 使用 gcc 命令将 .o 文件链接成最终的可执行文件 main
%.o : %.c # 任何 .c 文件都会生成对应的 .o 文件
gcc -c $< # 使用 gcc 编译 .c文件为.o文件,不进行链接,$<代表当前的.c文件
.PHONY: clean # 声明 clean 为伪目标,告诉make,clean不是文件目标,而是一个任务
clean: # clean 目标用于删除编译生成的文件
rm *.o # 删除所有 .o 文件
rm main # 删除可执行文件 main
正常编译和执行!
接下来,我们看看用到了哪些基本语法!
1. Makefile 规则基本格式
每一条 Makefile 规则通常遵循以下格式:
目标: 依赖文件
命令
命令
...
- 目标 (Target):是我们想要生成的文件,或者是一个抽象的操作(如
clean
)。 - 依赖文件 (Dependencies):在生成目标文件之前,需要先更新这些文件。如果依赖文件被修改过,make 会根据规则更新目标。
- 命令 (Command):用于生成目标的命令,通常是调用编译器(如
gcc
)或其他工具。命令必须以 TAB 键开始,而不能使用空格。
我们第一次编写时,一共有五条规则:
# 第一条规则:生成 main 目标
main: main.o input.o calcu.o # 目标是 main,依赖文件是 main.o, input.o 和 calcu.o
gcc -o main main.o input.o calcu.o # 如果任何 .o 文件更新,执行此命令生成 main 可执行文件
# 第二条规则:生成 main.o 目标
main.o: main.c # 目标是 main.o,依赖文件是 main.c
gcc -c main.c # 如果 main.c 更新,执行此命令生成 main.o
# 第三条规则:生成 input.o 目标
input.o: input.c # 目标是 input.o,依赖文件是 input.c
gcc -c input.c # 如果 input.c 更新,执行此命令生成 input.o
# 第四条规则:生成 calcu.o 目标
calcu.o: calcu.c # 目标是 calcu.o,依赖文件是 calcu.c
gcc -c calcu.c # 如果 calcu.c 更新,执行此命令生成 calcu.o
# 第五条规则:清理目标文件
clean: # clean 目标没有依赖文件,清理构建文件
rm *.o # 删除所有 .o 文件
rm main # 删除最终生成的可执行文件 main
在 Makefile
中,终极目标是默认构建的目标,通常是第一个规则的目标。当没有指定目标时,make
会自动执行第一个规则并构建相关依赖文件。但是这样写太麻烦了太麻烦了!!!!!
2. Makefile 变量
在Makefile
中,变量用于简化重复的代码,特别是在复杂的Makefile
中,避免重复的内容和减少错误。Makefile
中的变量本质上是字符串,可以在规则中动态引用。
main: main.o input.o calcu.o
gcc -o main main.o input.o calcu.o
main.o: main.c
gcc -c main.c
通过引入变量,Makefile 可以避免重复书写文件列表,提高可维护性。我们将 main.o
、input.o
和 calcu.o
这三个依赖文件提取到变量 objects
中,简化了后续的规则书写。下面是通过使用变量优化后:
objects = main.o input.o calcu.o #我们定义了一个变量 objects,其值为 main.o、input.o、calcu.o
main: $(objects) #目标 main 依赖于 $(objects),即main.o、input.o、calcu.o
gcc -o main $(objects) #使用 gcc 编译并链接所有的目标文件
我们使用到的变量objects
, Makefile
中变量的引用方法是$(变量名)
,比如本例中的$(objects)
就是使用变量objects
。当项目中的依赖文件很多时,使用变量可以大大简化Makefile
,提高可读性和可维护性。
变量赋值符
以下是关于 Makefile 变量赋值符号的表格,包含每种赋值符号的描述、用法和示例:
好的,这里是修改后的表格,只保留前两列:赋值符号和描述。
赋值符号 | 描述 |
---|---|
= | 延迟赋值,后续赋值会影响引用的值。 |
:= | 立即赋值,赋值时就固定,不受后续影响。 |
?= | 仅当变量未赋值时才赋予默认值。 |
+= | 向已有变量追加内容。 |
具体例子
(1)=
(延迟赋值):
A = 10 # 给变量 A 赋值为 10
B = $(A) # 给变量 B 赋值为 A 的值,即 10
A = 20 # 后续将 A 的值改为 20
print:
@echo B: $(B) # 输出 B: 20, 因为 B 引用了 A,A 的最终值是 20
输出:
B: 20
当使用 $(A)
引用变量时,它会引用变量定义时的最终值。后续的赋值会影响到所有引用它的变量。
(2):=
(立即赋值):
A := 10 # 立即赋值,A 的值固定为 10
B := $(A) # 立即赋值,B 的值为 A 的当前值 10
A = 20 # 修改 A 的值,不影响 B
print:
@echo B: $(B) # 输出 B: 10, 因为 B 在赋值时就固定了 A 的值为 10
输出:
B: 10
赋值后,变量的值是固定的,即使后续对原变量的值进行修改,也不会影响已经赋值的变量。
(3)?=
(条件赋值):
A ?= 10 # 如果 A 之前没有赋值过,则赋值为 10
print:
@echo A: $(A) # 输出 A: 10
输出:
A: 10
只有在变量未赋值时才会进行赋值。如果变量已经被赋值,?=
就不会有任何效果。
(4)+=
(追加赋值):
A = 10 # 给 A 赋值为 10
A += 20 # 向 A 追加 20
print:
@echo A: $(A) # 输出 A: 10 20,表示将 20 添加到 A 的原值后
输出:
A: 10 20
用于将新内容追加到已有的变量值中。
3. Makefile模式规则
在原始的 Makefile 中,为了编译每个 .c
文件,我们需要为每个 .c
文件写一条编译规则。比如:
main.o: main.c
input.o: input.c
calcu.o: calcu.c
模式规则允许我们用一种通配符方式为所有 .c
文件定义一个通用的规则。可以写成:
%.o : %.c
这里:
%.o
表示所有以.o
结尾的目标文件。%.c
表示与之对应的.c
源文件。
%.o : %.c
会为每个对应的 .c
文件生成一个 .o
文件,make
会根据目标 objects
中的每个文件,依次调用对应的编译规则,编译生成 .o
文件。所以如果有 main.c
、input.c
和 calcu.c
,输出的 .o
文件是 main.o
、input.o
和 calcu.o
。
那它怎么知道我有哪些文件呢?(我们是手动指定)
- 自动推导依赖:
Make
会查看当前目录下的所有.c
文件,并根据模式规则自动推导出对应的.o
文件。例如:假设你有main.c
、input.c
和calcu.c
这几个源文件,Make
会自动推导出main.o
、input.o
和calcu.o
作为目标文件。 - 手动指定对象文件:你还需要在 Makefile 中定义目标文件,即
.o
文件,来告诉Make
要编译哪些对象文件。你可以使用一个变量来列出所有目标文件(例如objects = main.o input.o calcu.o
)。
4. Makefile自动化变量
模式规则 允许我们用一个规则处理多个文件。那么自动化变量 会自动根据目标和依赖文件替换成相应的文件名,避免手动写出每个文件的名称。
gcc -c main.c
gcc -c input.c
gcc -c calcu.c
就可简化为:
gcc -c $<
每次执行规则 %.o : %.c
时,$<
只指代当前目标对应的 .c
文件。每个文件也会被逐个处理,生成对应的 .o
文件。这就是 make
的递归行为,它会针对每个目标和依赖文件执行相应的命令。
常用的自动化变量如下表:
自动化变量 | 描述 |
---|---|
$@ | 规则中的目标文件名。如果目标是一个函数库时,表示目标集合;在模式规则中,如果有多个目标文件,$@ 表示匹配模式中的目标文件。 |
$% | 仅当目标是一个函数库文件时,表示目标成员名。如果目标不是函数库文件,则为空。 |
$< | 规则中的第一个依赖文件名。如果依赖文件中使用了模式(如 %.c ),则 $< 表示符合模式的第一个文件。 |
$? | 所有比目标新的依赖文件集合,以空格分开。如果依赖文件较新,会列出这些依赖文件。 |
$^ | 规则中的所有依赖文件集合,以空格分开。如果依赖文件中有重复项,$^ 会去除重复文件。 |
$+ | 类似于 $^ ,但不会去除依赖文件中的重复项。 |
$* | 表示目标文件中,匹配模式中的 % 及其之前的部分。例如,目标文件是 test/a.test.c ,目标模式是 a.%.c ,那么 $* 就是 test/a.test 。 |
以上 7 个自动化变量中,常用的三种:$@
、$<
和$^
。
5. Makefile伪目标
假设你想在 Makefile
中写一个 clean
规则,用来删除 .o
文件和可执行文件。但是你项目中已经有一个名为 clean
的文件,make
可能会误认为你要生成一个叫 clean
的文件,而不是执行清理操作。
这时我们就可以使用 伪目标 来避免这种冲突,告诉 make
这个 clean
只是一个任务,而不是一个文件。
clean:
rm *.o
rm main
加上”伪目标标记“
为了告诉 make
某个目标是伪目标,你可以用 .PHONY
来声明它。这样即使有同名文件,make
也不会把它当作文件来处理。
.PHONY : clean
clean:
rm *.o
rm main
成功执行!!
十二、shell脚本入门
1. 什么是shell脚本
Shell 脚本是一种将一系列命令写入文件并一次性运行的技术。它可以简化繁琐的命令操作,特别适合需要频繁执行的任务。类似于 Windows 的批处理文件(.bat
),Shell 脚本不仅能运行命令,还支持数组、循环、条件判断等编程结构。简单来说,Shell脚本就是把多条命令写进一个文件,直接运行文件,就能一次性执行这些命令。
2. shell脚本写法
文件类型:Shell 脚本是一个纯文本文件,通常以 .sh
作为扩展名。
第一行声明:脚本的第一行必须指定解释器,例如:
#!/bin/bash
表示使用 Bash 解释器运行脚本。
3. shell脚本语法
3.1 第一个shell脚本
创建一个my.sh
文件,写入:
#!/bin/bash
echo "Hello, World!"
运行方式:
./my.sh
但是,运行时候发现没有权限!
chmod u+x my.sh # 给拥有者添加执行权限
3.2 交互式 Shell 脚本
通过 read
命令实现用户输入。
创建一个my1.sh
文件,写入:
#!/bin/bash
echo "请输入你的名字:"
read name
echo "你好,$name!"
chmod u+x my1.sh
./my1.sh
成功执行!
也可以同时多个输入,-p
用来在同一行显示提示信息,让用户输入更直观。
例如:
#!/bin/bash
read -p "Input your age and height: " age height
echo "Your age=${age}, your height=${height}"
如果输入 25 180
,脚本会输出:
Your age=25, your height=180
3.3 Shell 脚本的数值计算
Shell 仅支持整数运算,使用 $((表达式))
:
#!/bin/bash
a=10
b=20
sum=$((a + b))
echo "两数之和是:$sum"
chmod u+x my2.sh
./my2.sh
成功执行!
输入两个数值计算:
#!/bin/bash
read -p "请输入第一个数: " a
read -p "请输入第二个数: " b
sum=$((a + b))
echo "两数之和是:$sum"
3.4 test
命令
test
可用于检查文件状态、数值和字符串比较。例如:
文件测试:
test -e file.txt && echo "文件存在" || echo "文件不存在"
常用的test
命令如下表
类型 | 运算符/命令 | 用法示例 | 说明 |
---|---|---|---|
文件测试 | -e | test -e file.txt | 检查文件是否存在 |
-d | test -d dir | 检查是否为目录 | |
-r | test -r file.txt | 检查文件是否可读 | |
-w | test -w file.txt | 检查文件是否可写 | |
数值比较 | -eq | test $a -eq $b | 检查 $a 是否等于 $b |
-ne | test $a -ne $b | 检查 $a 是否不等于 $b | |
-gt | test $a -gt $b | 检查 $a 是否大于 $b | |
-lt | test $a -lt $b | 检查 $a 是否小于 $b | |
-ge | test $a -ge $b | 检查 $a 是否大于等于 $b | |
-le | test $a -le $b | 检查 $a 是否小于等于 $b | |
字符串比较 | = | test "$str1" = "$str2" | 检查字符串是否相等 |
!= | test "$str1" != "$str2" | 检查字符串是否不相等 |
3.5 &&
和 ||
运算符
&&
和 xx
运算符总结表格
运算符 | 语法 | 执行逻辑 |
---|---|---|
&& | cmd1 && cmd2 | 如果 cmd1 执行成功(返回值为 0 ),执行 cmd2 ;反之,不执行 cmd2 |
` | ` |
例子:
-
&&
:条件为真时执行后续命令:mkdir newdir && cd newdir
-
||
:条件为假时执行后续命令:test -e file.txt || echo "文件不存在"
3.6 中括号 [ ]
判断符
运算符 | 语法 | 说明 |
---|---|---|
= | [ "$str1" = "$str2" ] | 判断两个字符串是否相等 |
!= | [ "$str1" != "$str2" ] | 判断两个字符串是否不相等 |
[ "$str1" = "$str2" ]
:判断两个字符串是否相等。如果相等,返回真(0),否则返回假(1)。[ "$str1" != "$str2" ]
:判断两个字符串是否不相等。如果不相等,返回真(0),否则返回假(1)。
例子:
#!/bin/bash
str1="hello"
str2="world"
if [ "$str1" = "$str2" ]; then
echo "字符串相等"
else
echo "字符串不相等"
fi
输出:
字符串不相等
3.7 默认变量
变量 | 说明 | 示例 | 解释 |
---|---|---|---|
$0 | 脚本名称 | echo $0 | 显示脚本的文件名。 |
$1 至 $n | 脚本参数,第 1 至第 n 个 | echo $1 | 显示第 1 个参数(如:$1=arg1 )。 |
$# | 参数个数 | echo $# | 显示传入的参数个数。 |
$@ | 所有参数(保留引号) | echo "$@" | 显示所有传入的参数。 |
例子:
#!/bin/bash
echo "脚本名:$0"
echo "第一个参数:$1"
echo "参数个数:$#"
echo "所有参数:$@"
执行:
./myscript.sh arg1 arg2 arg3
输出:
脚本名:./myscript.sh
第一个参数:arg1
参数个数:3
所有参数:arg1 arg2 arg3
十三、shell脚本条件判断、函数和循环
下面是对 Shell 脚本条件判断、函数和循环 的精简总结,包括常用的语法和示例:
1. Shell 脚本条件判断
if
then
判断
if 条件判断; then
# 条件成立时执行的代码
fi
if
then
else
判断
if 条件判断; then
# 条件成立时执行的代码
else
# 条件不成立时执行的代码
fi
if
then
elif
else
判断
if 条件判断; then
# 条件成立时执行的代码
elif 条件判断; then
# 第二个条件成立时执行的代码
else
# 条件都不成立时执行的代码
fi
case
语句
case $变量 in
"第1个变量内容")
# 执行代码
;;
"第2个变量内容")
# 执行代码
;;
"第n个变量内容")
# 执行代码
;;
esac
例子:
2. Shell 脚本函数
- 函数定义
function fname() {
# 函数代码段
}
fname # 调用函数
例子:
3. Shell 脚本循环
while
循环:条件成立时持续执行
while [ 条件 ]; do
# 循环代码
done
until
循环:条件不成立时持续执行
until [ 条件 ]; do
# 循环代码
done
for
循环:逐个遍历列表
for var in 1 2 3 4; do
# 循环代码
done
for
循环(数值范围)
for (( 初始值; 限制值; 步长 )); do
# 循环代码
done
例子:
总结
自用