当前位置: 首页 > article >正文

【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 和其他操作系统有什么区别?

特点LinuxWindowsmacOS
价格免费,人人可用。需要花钱买授权(不是免费的)。买苹果电脑时自带系统。
自由性可以随意修改和定制。系统固定,不能修改。系统固定,不能修改。
安全性病毒少,很安全。病毒多,需要杀毒软件保护。安全性高,病毒很少。
谁在用程序员、技术人员、服务器管理员。普通用户、企业、游戏玩家。设计师、视频编辑、苹果用户。
运行速度很快,对旧电脑也友好。速度一般,需要好配置的电脑。优化好,速度快,但只适配苹果。
软件支持免费工具多,商用软件和游戏较少。商业软件和游戏最多。设计和创意类软件丰富。

一、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存放基本系统命令,如 catcpmkdir 等。用于系统启动和修复。
/boot存放开机时需要的文件,如启动管理程序 grub2
/dev存放设备文件,用于与硬件设备(如声卡、硬盘、光驱等)进行交互。
/etc系统配置文件的存放目录,包含网络配置、用户配置等。
/home用户的家目录,存放普通用户的个人文件。
/lib存放系统命令(binsbin 目录下的命令)所需的共享库文件。
/lib32//lib64存放 32 位和 64 位系统所需的二进制共享库文件。
/lost+found用于存放系统崩溃或关机时丢失的文件碎片,系统修复时会检查并恢复这些文件。
/media用于挂载可移动设备,如 CD、DVD、U盘等。
/mnt用于临时挂载存储设备,通常由管理员手动挂载设备。
/opt存放第三方软件安装目录。
/proc存放进程信息和内核信息,提供系统运行时的数据,不占用硬盘空间。
/rootroot 用户的家目录。
/run存储系统启动以来的信息,系统重启后会清空。
/sbin存放系统管理员(root)使用的命令,如磁盘管理、网络管理等。
/srv存放服务相关的数据文件,如 Web 服务、FTP 服务的数据。
/sys存放与系统硬件相关的信息,类似于 proc,用于内核与硬件之间的通信。
/tmp存放临时文件,程序运行时产生的临时数据,系统重启后通常会清空。
/usr存放系统程序和库文件,类似于 Windows 系统中的 Program Files 目录。
/var存放内容变化频繁的文件,如日志文件、缓存文件、邮件等。

绝对路径和相对路径

特性绝对路径相对路径
起始点从根目录 / 开始,表示完整路径。从当前工作目录开始,表示相对于当前位置的路径。
访问方式无论当前目录在哪里,路径始终指向同一个目标。路径取决于当前所在的目录。
表示方式必须以 / 开头,包含了从根目录开始的所有目录。不以 / 开头,只描述相对位置。
示例/home/user/documents/file.txtdocuments/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不保存退出

分区操作示例

  1. 启动 fdisk 分区工具:

    sudo fdisk /dev/sdb
    
  2. 输入 m 查看帮助,选择对应的操作:

    • 输入 n 创建新分区
    • 输入 w 保存更改

5. 格式化分区命令 mkfs

创建文件系统的命令。常用于格式化硬盘分区。

常用选项

选项功能
-t指定文件系统类型(如 vfat, ext4 等)
-v显示详细执行过程

格式化示例

  1. 格式化为 vfat 文件系统:

    sudo mkfs -t vfat /dev/sdb1
    
  2. 格式化为 ext4 文件系统:

    sudo mkfs -t ext4 /dev/sdb1
    

6. 挂载分区命令 mount

将硬盘分区挂载到指定目录,之后就可以通过该目录访问磁盘内容。

常用选项

选项功能
-t指定文件系统类型(如 vfat, ext4 等)
-o ro只读模式挂载
-o rw读写模式挂载
-v显示详细的执行过程

挂载示例

  1. 创建挂载点目录:

    sudo mkdir /mnt/tmp
    
  2. 挂载分区 /dev/sdb1/mnt/tmp

    sudo mount -t vfat /dev/sdb1 /mnt/tmp
    

7. 卸载分区命令 umount

当磁盘或分区不再需要时,使用 umount 命令将其从挂载点卸载。

常用选项

选项功能
-v显示详细信息
-r如果无法卸载,尝试以只读方式卸载
-a卸载 /etc/mtab 中的所有文件系统

卸载示例

  1. 卸载挂载点 /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/sdb1vfat 文件系统。
挂载磁盘分区sudo mount -t vfat /dev/sdb1 /mnt/tmp/dev/sdb1 挂载到 /mnt/tmp 目录。
卸载磁盘分区sudo umount /mnt/tmp卸载 /mnt/tmp 目录下的挂载分区。

四、Ubuntu下压缩与解压缩

【待更改】

Linux 常用压缩和解压缩工具

工具压缩命令解压命令描述
gzipgzip xxxgzip -d xxx.gz压缩和解压 .gz 格式文件
gzip -r xxxgzip -rd xxx.gz压缩/解压文件夹(递归)
bzip2bzip2 -z xxxbzip2 -d xxx.bz2压缩和解压 .bz2 格式文件
tartar -vcf test.tar testtar -vxf test.tar打包和解包文件(不含压缩)
tar -vxjf xxx.tar.bz2tar -vcjf xxx.tar.bz2 xxx压缩/解压 .tar.bz2 格式文件
tar -vxzf xxx.tar.gztar -vczf xxx.tar.gz xxx压缩/解压 .tar.gz 格式文件
rarrar a xxx.rar xxxrar x xxx.rar压缩和解压 .rar 格式文件
zipzip -rv xxx.zip xxxunzip -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 查看用户所属组
用户组与权限文件的访问权限可以基于用户组进行设置,组内用户共享文件访问权限chownchmod 设置文件所有权和权限

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 = 4w = 2x = 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 1210 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 1210 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 1210 12:04 a.c   #显示拥有者已有执行权
【输入】./a.c                                      #再次执行(无显示,就是可执行)

方式二:数字法

使用权限值修改权限,每组三位,分别表示 u, g, o 的权限:

  • 权限值为 rwx 的加和,r = 4w = 2x = 1,例如:rw- = 6r-- = 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 1210 12:04 a.c
【更改后】-rwxrw-r-- 1 root lty 0 1210 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.txtfile2.txt 其实是同一个文件,只是不同的名字。删除其中一个,另一个依然有效,文件内容不受影响。
  • 如果 file1.txt有10kb,创建了一个硬连接 file2.txt,那么 file1.txtfile2.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.txtlink_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,显示错误,但是我们代码是正确的,为什么呢?

错误原因

  1. 源代码文件不能直接执行
    • 虽然我们给 hello.c 文件添加了可执行权限,但它是一个源代码文件,而不是可执行程序,不能直接执行。直接运行源代码文件会导致 Bash 尝试将其作为脚本执行,从而产生语法错误。
  2. 错误信息解释
    • 错误信息 未预期的记号 "(" 附近有语法错误 表示 Bash 试图解析 C 语言的语法,但它并不理解 C 语言的语法结构。

正确的工作流程

正确运行 C 代码,要遵循以下步骤:

  1. 编译源代码

    • 使用 gcc 编译源代码文件 hello.c,生成可执行文件。
    gcc 需要编译的文件 -o 生成的可执行文件  
    
  2. 运行可执行文件

    • 运行生成的可执行文件 hello,而不是源代码文件 hello.c
    ./可执行的文件 
    

在这里插入图片描述

gcc hello.c -o hello
./hello

OK,终于成功运行了!

回归正题,将hello文件分别进行硬连接和软连接

4.1 硬连接

创建硬连接的命令:

ln [源文件] [硬连接文件]

我们创建两个硬连接hello1hello2

ln hello hello1
ln hello hello2

在这里插入图片描述

(1)硬连接和源文件共享同一个inode,什么是inode,怎么看呢?

简单来说,inode 就是文件的“身份证”,它记录了文件的基本信息,帮助操作系统管理文件。我们输出以下命令,用来列出所有以 hello 开头的文件,并显示它们的 inode 号码。

ll -i hello*
  • ll:是 ls -l 命令的简写,用于列出当前目录中所有文件和子目录的详细信息;
  • -i:用于显示文件的 inode号码;
  • hello\*:这是一个通配符,用于匹配所有以 hello 开头的文件或目录。例如,hello1hello2hello123 等都符合这个模式。

在这里插入图片描述

从输出可知,hellohello1hello2 这三个文件都指向相同的 inode(3028616),并且它们的链接数都变成了 3。这意味着我们已经创建了三个硬连接指向同一个文件。

(2)我们删掉一个硬连接,或者直接删掉源文件,看看其他的还能打开吗?

在这里插入图片描述

删掉其他的硬连接和源文件,其他的硬连接都能这些执行!只有所有硬连接删除后,文件内容才会被删除。

(3)那么如果我们改动一个硬连接的内容,其他的会跟着改变吗?

我们将hello.c创建一个硬连接hello1.c,和源文件内容一致

在这里插入图片描述

把硬连接hello1.c改动一下,点击保存

在这里插入图片描述

再打开hello.c

在这里插入图片描述

修改任意硬连接都会影响源文件和所有硬连接!!!

4.2 软连接

创建硬连接的命令:

ln -s [源文件] [软连接文件]

我们创建两个硬连接helloAhelloB

ln -s hello helloA
ln -s hello helloB

在这里插入图片描述

(1)软连接(符号链接)与硬连接不同,它有自己的 inode 和内容。

但它的 inode 并不直接存储文件的数据内容,而是存储指向目标文件路径的信息。

在这里插入图片描述

(2)删掉一个软连接,或者删掉源文件,看看其他软连接还能正常执行吗?

在这里插入图片描述

删除软连接本身并不会影响源文件或其他软连接。软连接只是一个指向目标文件路径的快捷方式,它与目标文件是独立的。其他指向同一源文件的软连接仍然有效,能够正常执行。

如果你删除了源文件,软连接会失效,指向的路径变得无效。所有指向该文件的软连接将变为悬挂链接。这些软连接将无法访问源文件内容,因为源文件已经不存在了。

(3)那么如果我们改动一个软连接的内容,其他的会跟着改变吗?

我们将hello.c创建一个软连接helloA.c,和源文件内容一致

在这里插入图片描述

把软连接helloA.c改动一下,点击保存

在这里插入图片描述

再打开hello.c

在这里插入图片描述

修改任意软连接都会影响源文件和所有软连接!!!

(4)符号连接(软连接) 推荐使用绝对路径,为什么呢?

我们先创建两个文件夹testtest1,将上面的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.outgcc -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 命令来编译;但如果文件数目增多(如几十、上百甚至上千个文件),使用终端命令就不再方便。为了解决这个问题,我们就可以使用 Makefilemake 工具。

十、make工具和Makefile的引入

make工具
make是一个“智能”的编译助手。在编程时,我们通常会编写多个源文件,并希望将它们编译成一个可执行程序。make能够自动找出哪些源文件已经改变,并仅重新编译这些改变的文件,而不是全部重新编译。这样,可以大大节省编译时间。

Makefile
Makefile是一个配置文件,它告诉make如何编译程序。在这个文件中,我们列出了要编译的源文件、编译时需要的选项、以及源文件之间的依赖关系等。当我们在命令行中运行make命令时,make会读取Makefile中的信息,并按照指定的规则执行编译任务。

1. 具体例子

接下来,我们写一个多个文件的代码,试试看普通编译和使用make/makefile有什么区别!

首先,我们将构建一个简单的C程序,包含三个源文件 main.c, input.c, calcu.c,和两个头文件 input.hcalcu.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.oinput.ocalcu.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 编译并链接所有的目标文件

我们使用到的变量objectsMakefile中变量的引用方法是$(变量名),比如本例中的$(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.cinput.ccalcu.c,输出的 .o 文件是 main.oinput.ocalcu.o

那它怎么知道我有哪些文件呢?(我们是手动指定)

  • 自动推导依赖Make 会查看当前目录下的所有 .c 文件,并根据模式规则自动推导出对应的 .o 文件。例如:假设你有 main.cinput.ccalcu.c 这几个源文件,Make 会自动推导出 main.oinput.ocalcu.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命令如下表

类型运算符/命令用法示例说明
文件测试-etest -e file.txt检查文件是否存在
-dtest -d dir检查是否为目录
-rtest -r file.txt检查文件是否可读
-wtest -w file.txt检查文件是否可写
数值比较-eqtest $a -eq $b检查 $a 是否等于 $b
-netest $a -ne $b检查 $a 是否不等于 $b
-gttest $a -gt $b检查 $a 是否大于 $b
-lttest $a -lt $b检查 $a 是否小于 $b
-getest $a -ge $b检查 $a 是否大于等于 $b
-letest $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

例子:在这里插入图片描述在这里插入图片描述


总结

自用


http://www.kler.cn/a/442728.html

相关文章:

  • SpringBoot整合Easy-es
  • 一个使用 Golang 编写的新一代网络爬虫框架,支持JS动态内容爬取
  • 如何选择Ubuntu版本
  • 后端技术选型 sa-token校验学习 下 结合项目学习 后端鉴权
  • 如何开放2375和2376端口供Docker daemon监听
  • 源码编译安装httpd 2.4,提供系统服务管理脚本并测试(两种方法实现)
  • 画图,matlab,
  • 搭建MongoDB
  • 渗透测试实战—利用防火墙突破网络隔离
  • vue3的element的日期组件添加prefix
  • 【MySQL】数据库必考知识点:查询操作全面详解与深度解剖
  • Halcon中split_skeleton_lines(Operator)算子原理及应用详解
  • Springboot 启动 Property ‘sqlSessionFactory‘ or ‘sqlSessionTemplate‘ are required
  • DC-6笔记
  • Mysql学习笔记之SQL-2
  • 剑指Offer|LCR 007. 三数之和
  • sunset: midnight
  • Elasticsearch Kibana (windows版本) 安装和启动
  • vue3-tp8-Element:对话框实现
  • TCP Analysis Flags 之 TCP Fast Retransmission
  • 【Unity功能集】TextureShop纹理工坊(二)图层(下)
  • 车辆重识别代码笔记12.18
  • JS的原型和原型链浅析
  • 深度学习中,卷积层的若干思考!!!
  • 【OSS】php使用oss存储
  • 【Elasticsearch】使用阿里云 infererence API 及 semantic text 进行向量搜索