Linux shell脚本教程

作者:诸神缄默不语日期:2026/6/4

诸神缄默不语-个人技术博文与视频目录

Linux系统的命令行终端界面就是一个小黑窗,在里面敲命令执行任务。当你想执行一系列复杂的任务(比如连续执行多个命令、有逻辑判断规则等)时,光靠直接敲命令+回车就不够了,这时你就会将一系列任务的执行代码写到一个文本文件中,然后让Linux终端依次执行。这个文本文件就是shell脚本。
本文对Linux系统中的shell脚本进行简单介绍,包括其作用和基本写法。更高级的用法将在以后的教程中介绍。
对Linux系统的整体命令行操作教程,请参考我撰写的另一篇博文:Linux命令行教程-CSDN博客

文章目录

  • 1. shell脚本简单示例
  • 2. Linux系统中的一些基础概念
    • 1. shell
    • 2. shell语言
    • 3. bash
      • 1. 名字的来源:为什么叫 bash?
        • 2. bash 和 sh 是什么关系?
        • 3. bash 能干什么?
        • 4. 一个小对比,帮你理解 bash 的“增强”之处
    • 4. shebang
      • 1. 语法形式
        • 2. 它是怎么工作的?
        • 3. 写法
        • 4. 没有 Shebang 会怎样?
        • 5. shebang一词由来
    • 5. bash的配置文件:`~/.bashrc` 和 `~/.bash_profile`
      • 1. `.bash_profile` — 登录时执行一次
        • 2. `.bashrc` — 每次打开新 shell 都执行
        • 3. bash配置文件启动时机图示
        • 4. 还有 `.profile` 是什么?
        • 5. 修改后如何生效?
        • 6. 一个典型的配置示例
  • 3. Shell脚本
    • 1. 脚本的基本结构
    • 2. 创建与运行
    • 3. 核心语法速览
    • 4. 常见用途

1. shell脚本简单示例

backup.sh:执行备份,把当前目录压缩并带上日期

1#!/bin/bash
2date_str=$(date +%Y%m%d)
3target="backup_${date_str}.tar.gz"
4
5tar -czf "$target" /path/to/source
6echo "Backup created: $target"
7

将文本文件复制到Linux系统工作路径下,赋予执行权限后运行:

1chmod +x backup.sh
2./backup.sh
3

2. Linux系统中的一些基础概念

1. shell

Shell 的字面意思是“壳”。在计算机里,它和“内核(Kernel)”相对,指包裹在操作系统内核外面的一层命令行解释器,是用户和系统之间的接口,负责读取你输入的命令,翻译后交给内核执行,再把结果返回给你。

Shell = 命令语言解释器 + 脚本编程语言 + 工作环境

Shell 全景图:

1你(用户)
2   输入命令
3[ Shell ]    sh, bash, zsh, fish ... 任你选
4   系统调用
5[ 内核 (Kernel) ]  管理 CPU、内存、磁盘
6   结果返回
7[ Shell ]  显示在屏幕上让你看到
8

Shell 的两种使用方式:

  • 交互式:你打开终端,看到 $ 提示符,输入一条命令,回车,立即看到结果。
1$ ls -l
2$ echo "hello"
3
  • 非交互式(脚本):把一堆命令写进文件(.sh),一次性顺序执行,就是我们一直在说的 Shell 脚本

Shell 有很多种实现:

Shell 名称路径特点
sh (Bourne Shell)/bin/sh最早的 Unix Shell,所有 Shell 的老祖宗
bash (Bourne Again SHell)/bin/bashsh 的增强版,Linux 默认,功能最丰富
dash/bin/dash轻量级 sh,启动快,常用于系统脚本
zsh/bin/zsh兼容 bash,更强大的主题和插件系统(macOS 默认)
fish/usr/bin/fish开箱即用的友好交互,但语法不兼容 bash
kshtcsh各自路径各有特色

2. shell语言

Shell 提供了变量、条件判断、循环、函数等编程结构,所以你可以在里面写逻辑。

一个最简单的 Shell 脚本(不管用哪个 Shell,基本写法都类似):

1#!/bin/sh
2for i in 1 2 3; do
3    echo "Count: $i"
4done
5

这里的 #!/bin/sh 就是 shebang,告诉系统:“请用 /bin/sh 这个 Shell 来执行本脚本”。你换成 #!/bin/bash 就会用 Bash 执行,可以使用 Bash 独有的增强功能。

3. bash

bash 是一个 Shell(壳),全称是 Bourne Again SHell,它是绝大多数 Linux 发行版和 macOS 的默认命令行解释器

1. 名字的来源:为什么叫 bash?

它是由 Brian Fox 在 1987 年为 GNU 项目编写的,目的是替代 Unix 上原有的 Bourne shell(sh

  • Bourne:原版的 sh 作者叫 Stephen Bourne
  • Again:意味着重生、再版
  • 连起来就是 Bourne Again SHell,同时也故意与“born again(重生)”谐音

2. bash 和 sh 是什么关系?

sh (Bourne shell)bash (Bourne Again SHell)
发布时间1977 年1989 年
命令补全不支持支持(按 Tab 键)
命令历史支持上下箭头、history 命令
脚本语法基础 if/for/case兼容 sh,并增加 [[、数组、函数增强等
默认路径/bin/sh/bin/bash

在很多现代 Linux 系统中,/bin/sh 实际上已经是 /bin/bash 的链接(或一个简化模式的 bash),但为了脚本可移植性,严谨的脚本还是会区分 #!/bin/sh(保证符合 POSIX 标准)和 #!/bin/bash(可以使用 Bash 专有的扩展功能)。

3. bash 能干什么?

你与 Linux 交互的绝大多数场景都在使用 bash(或其他 shell):

  • 交互式命令行:打开终端,看到 user@host:~$ 提示符,背后就是 bash 在等你输入。
  • 执行脚本:你的 .sh 脚本第一行写 #!/bin/bash,就是在告诉系统:“请用 bash 解释器来跑这个文件”。
  • 定制环境:启动时自动加载 ~/.bashrc~/.bash_profile,为你配置别名、环境变量等。
  • 任务自动化:把一堆命令写成 bash 脚本,一键完成备份、部署、数据处理。

4. 一个小对比,帮你理解 bash 的“增强”之处

老式 sh 的写法

1if [ "$name" = "hello" ]; then
2   echo "ok"
3fi
4

bash 支持的增强写法(更安全、功能更全):

1if [[ $name == hello ]]; then
2   echo "ok"
3fi
4

这里 [[ ]] 是 bash 的关键字,比 [ ] 更强大,比如支持正则、不需要对变量加双引号防出错等。这就是为什么很多脚本明确要求 #!/bin/bash

4. shebang

Shebang(也叫 #! 行)是脚本文件的第一行,用来告诉操作系统:“请用这个解释器来执行此文件”。

它只对直接执行脚本生效(./script),如果用 bash script.sh python script.py 显式调用解释器,shebang 会被忽略。

1. 语法形式

1#!解释器路径 [可选参数]
2
  • #! 必须出现在文件的第一行最开头,前面不能有空行或空格。
  • 后面紧跟解释器的绝对路径,可以带一个可选参数(但不同系统支持程度不同)。

2. 它是怎么工作的?

当你直接运行一个脚本(如 ./script.sh)时,内核会读取文件开头,如果前两个字节是 #!,就会提取后面指定的解释器路径,然后变成类似这样的调用:

1解释器路径 脚本名 [你传入的其他参数]
2

例如,一个 #!/bin/bash 的脚本 myscript.sh,运行 ./myscript.sh hello 时,内核实际执行:

1/bin/bash ./myscript.sh hello
2

所以文件本身不需要执行权限吗?仍然需要,因为内核先要能“执行”这个文件,才会去读它的 shebang。

3. 写法

示例:

1#!/bin/bash            # Bash 脚本
2#!/bin/sh              # 系统默认 sh
3#!/usr/bin/python3     # Python 3 脚本(详见下文)
4

两种写法:硬路径 vs 环境变量路径

  • 直接写死路径#!/bin/bash
    优点:精确,不受环境变量影响。
    缺点:如果目标系统解释器不在该路径(如某些 Linux 上 bash 在 /usr/bin/bash),脚本会因“找不到解释器”而失败。
  • 使用 /usr/bin/env#!/usr/bin/env python3
    这时 env 会在 PATH 环境变量里搜索 python3,并调用找到的第一个。
    优点:跨平台性更好,不用猜测解释器装在哪里。
    缺点:会依赖调用者的 PATH,如果 PATH 被篡改,可能找到错误的解释器。不过日常使用很广泛。

注意:env 方式通常只接受一个解释器名,不能带额外参数(像 #!/usr/bin/env python3 -u 在一些老系统上可能不支持)。

值得注意的是,这里Python脚本虽然可以这样写shebang,但是事实上现在更流行的运行Python脚本的方法是先激活虚拟环境,再直接用python跑脚本:

1# 先激活虚拟环境
2source venv/bin/activate   #  conda activate myenv
3# 然后直接用 python 跑脚本
4python myscript.py  #  uv run script.py
5

这种情况下会直接忽略shebang,因为已经指定了执行解释器(python)。如果你想直接像脚本一样执行Python代码,就加shebang;而且pip/conda安装的第三方包里有些也会显示加上#!/usr/bin/env python或者类似的shebang,这个shebang就用的本地python虚拟环境了。

4. 没有 Shebang 会怎样?

如果脚本没有 shebang,并且你尝试直接 ./script.sh 执行,系统通常会用当前用户默认的 shell(一般为 /bin/sh)来执行。这可能导致用 bash 语法写的脚本在 sh 下出错。所以建议始终写明 shebang,即使你只计划用 bash script.sh 手动运行,那样会指定执行解释器,就不会使用shebang——良好的 shebang 能让脚本自描述。

5. shebang一词由来

“Shebang” 是 #! 两个字符的俚语:# 叫 “sharp” 或 “hash”,! 叫 “bang”,合起来就是 sharp-bang → shebang。有时也叫 hash-bang。

1#!/bin/bash
2   ^     ^
3   |     |
4  hash  bang   "shebang"
5   |
6告诉系统:
7 /bin/bash 执行这个脚本
8

5. bash的配置文件:~/.bashrc~/.bash_profile

这两个文件本身也是脚本,但是不是只需要执行一次的脚本,而是希望在每次登录shell或者开启新终端时都要先执行一次。比如写入环境变量,或者开启Python虚拟环境等任务。

1. .bash_profile — 登录时执行一次

当你在以下场景登录时,Bash 会作为 login shell 启动,并读取 ~/.bash_profile(或 ~/.bash_login~/.profile):

  • 通过 SSH 远程登录服务器
  • 在虚拟控制台(Ctrl+Alt+F1~F6)登录
  • 使用 su - username 切换用户
  • macOS 的终端默认以 login shell 启动

用途:适合放一些只需要执行一次的环境初始化,例如:

惯例.bash_profile 的最后通常会显式调用 .bashrc,以避免配置重复:

1if [ -f ~/.bashrc ]; then
2    source ~/.bashrc
3fi
4

2. .bashrc — 每次打开新 shell 都执行

当你打开一个非登录的交互式 shell(interactive non-login shell)时,Bash 会读取 ~/.bashrc。这包括:

  • 在图形界面(Gnome、KDE)中打开一个新终端窗口
  • 在已有终端里直接输入 bash 启动子 shell
  • 在 VS Code 等编辑器的集成终端中打开

用途:放那些每次打开终端都要生效的设置,例如:

  • 命令别名(alias ll='ls -la'
  • 自定义函数
  • Shell 提示符(PS1)美化
  • 命令补全设置、历史记录控制

3. bash配置文件启动时机图示

1登录(login shell)
2   读取 ~/.bash_profile
3      ├─ 设置环境变量
4      └─ source ~/.bashrc    这里调用 .bashrc
5           ├─ 设置别名
6           └─ 设置提示符
7
8之后每打开一个新终端(non-login shell)
9   直接读取 ~/.bashrc    不再读 .bash_profile
10

这样设计是为了避免重复执行:环境变量只需设置一次,而别名、提示符等需要每个 shell 实例都生效。

4. 还有 .profile 是什么?

如果 ~/.bash_profile 不存在,Bash 会查找 ~/.bash_login,再没有就找 ~/.profile(这是通用 Shell 配置,为兼容 sh 保留)。通常建议只用 .bash_profile + .bashrc 组合

5. 修改后如何生效?

  • 想让 .bashrc 在当前终端生效:source ~/.bashrc. ~/.bashrc
  • 想让 .bash_profile 生效:重新登录,或执行 source ~/.bash_profile(但环境变量可能会传递给子进程,建议重新登录更干净)

6. 一个典型的配置示例

~/.bash_profile

1# 只做登录时的初始化
2export EDITOR=vim
3export PATH="$HOME/bin:$PATH"
4
5# 载入 bashrc
6if [ -f ~/.bashrc ]; then
7    . ~/.bashrc
8fi
9

~/.bashrc

1# 每次新 shell 都要的配置
2alias ll='ls -alF'
3alias gs='git status'
4
5PS1='\u@\h:\w\$ '   # 提示符格式
6
7# 历史记录设置
8export HISTSIZE=1000
9export HISTFILESIZE=2000
10

这样,无论登录还是新开窗口,你都能获得一致且舒适的 Shell 环境。

3. Shell脚本

Shell 是用户与系统内核之间的命令行解释器(比如 bashzsh)。脚本就是把要敲的命令保存成文件,由 Shell 逐行解释执行,省去重复输入的麻烦。

常见后缀
(事实上可以使用任意后缀,或无后缀,因为Linux 不依赖后缀识别解释器,靠 shebang 识别。后缀只是为了便于人类理解)

后缀对应解释器说明
.sh通常指 sh 或 bash最常见的脚本后缀,约定俗成
.bashbash明确使用 Bash 特性时用
.cshcsh / tcshC Shell 语法,类似 C 语言
.zshzshZ Shell 脚本

1. 脚本的基本结构

一个规范的脚本至少包含两部分:

1#!/bin/bash
2# 这是注释,说明脚本功能
3echo "Hello, Linux!"
4
  • 第一行 #!/bin/bash:叫 shebang,告诉系统用哪个解释器执行(换成 #!/bin/sh#!/usr/bin/python3 则用相应解释器)。
  • 之后:写任何你会在终端输入的命令,一行一个。

2. 创建与运行

1# 1. 创建并编辑脚本
2vim my_script.sh
3
4# 2. 赋予执行权限
5chmod +x my_script.sh
6
7# 3. 运行
8./my_script.sh          # 需要执行权限
9# 
10bash my_script.sh       # 不需要执行权限,显式指定解释器
11
  • 直接 ./ 执行时,脚本里必须有 shebang 且文件有执行权限。
  • bash 运行会忽略 shebang,临时强制用 Bash 解释。

vim教程可参考我撰写的另一篇博文:在Linux中使用Vim编辑文本

3. 核心语法速览

变量

1name="world"
2echo "Hello, ${name}"   # 推荐用 ${} 包裹变量
3

特殊变量

  • $0:脚本名
  • $1, $2...:传入的参数
  • $#:参数个数
  • $?:上一条命令的退出状态(0 成功)

条件判断

1if [ "$name" == "world" ]; then
2    echo "matched"
3else
4    echo "not matched"
5fi
6

循环

1for i in 1 2 3; do
2    echo "Number $i"
3done
4
5# 或者读取文件行
6while read line; do
7    echo "$line"
8done < file.txt
9

函数

1greet() {
2    local msg="Hello, $1"  # local 定义局部变量
3    echo "$msg"
4}
5greet "Alice"
6

4. 常见用途

  • 自动化运维:批量重命名、备份数据库、部署服务
  • 定时任务:配合 cron 定期执行(比如每天凌晨清理日志)
  • 环境初始化:安装软件包、配置变量(类似 ~/.bashrc 做的事)
  • 胶水代码:把不同命令行工具串起来处理数据

Linux shell脚本教程》 是转载文章,点击查看原文


相关推荐


零基础webgis开发入门:HTML/CSS/JavaScript前端核心基础②
GIS6688002026/5/28

CSS:页面样式与布局美化 CSS 全称层叠样式表,核心作用是控制HTML元素的外观和布局,包括大小、颜色、背景、位置、边距等。 在WebGIS中,CSS直接决定地图的显示尺寸、是否全屏、页面是否有白边等关键效果。 CSS的核心逻辑可总结为两步走:选元素、改样式。 第一步:选元素(选择器) 1)什么是选择器: 选择器是CSS的核心,作用是从页面众多HTML元素中,筛选出需要修改样式的目标元素。 想象一群小黄人站你面前,你想把单眼的小黄人选出来变红色。 第一步:选出所


TOML 深度调研:对比 YAML、JSON 等五大配置格式,哪种最适合你的项目?
王若风2026/5/6

大家好,我是若风。 上周在配置一个 Rust 项目的时候,我盯着 Cargo.toml 发了一会儿呆。然后突然意识到一件事:我写了这么多年代码,跟配置文件打交道的时间可能比写业务逻辑还多。package.json、docker-compose.yml、tsconfig.json、.gitignore、terraform.tf……每个项目至少 3 到 5 个配置文件。 但说实话,我从来没认真想过一个问题:为什么这些工具要用不同的配置格式? YAML 写 Kubernetes 配置,JSON 写 p


如何将SVG格式文件转为PDF? 方便打印输出、正式汇报、跨平台展示
诸葛大钢铁2026/4/26

在日常设计、开发与文档交付过程中,SVG转PDF是一个非常高频但容易被忽视的需求。很多人一开始会觉得“只是格式转换而已”,但真正遇到输出打印、正式汇报或跨平台展示时才发现:SVG在不同设备上的兼容性并不总是稳定,而PDF才是更通用、更专业的交付格式。 尤其是在以下场景中,这个需求会变得非常明显: 设计稿需要提交评审或印刷 网页图标或流程图需要归档成标准文件 跨平台传输时避免样式错乱 因此,一个稳定、清晰、无损的SVG转PDF方案就显得非常重要。 一、设计软件直接导出


《swiftUI进阶 第9章SwiftUI 状态管理完全指南》
90后晨仔2026/4/18

概述 状态管理是 SwiftUI 应用的核心。本章将系统介绍从 iOS 13 到 iOS 17+ 的所有状态管理技术,包括传统的 ObservableObject 系列和现代的 @Observable 宏,帮助你根据项目需求选择最合适的方案。 第一部分:基础状态管理(iOS 13+) 1. @State:本地视图状态 @State 用于管理视图内部的简单状态,当值改变时自动刷新 UI。 struct CounterView: View { @State private var coun


Nginx 从入门到精通:全面解析与实战指南
程序员果子2026/4/10

目录 前言:为什么要学 Nginx? 一、Nginx 基础入门:从零搭建第一个服务 1.1 初识 Nginx:它是什么,能做什么? 1.2 第一个 Nginx 服务:最小化配置实战 1.3 安装:Linux 里的 Nginx 魔法:从下载到部署,轻松拿捏! 二、核心架构与配置解析:读懂 Nginx 的 "运行逻辑" 2.1 架构精髓:Master-Worker 进程模型 2.2 配置骨架:从 http 到 location 的层级关系 2.3 静态资源服务:Nginx 的 "原


多Agent工作流开发
字节逆旅2026/4/2

最近 OpenClaw、Claude Code 特别火,我当时就在想,能不能写个自己的Agent,让它自动根据我的需求文件干活?比如我改个 tasks.md,它就自动跳出来把代码写了。这不比怼着ide开发高级多了? 最开始的想法非常简单粗暴:用 Node.js 写个脚本,利用 chokidar 盯着一个 todo 文件夹。只要文件一变,脚本就通过 child_process 里的 exec 去调 claude 命令。 初版脚本核心逻辑: const chokidar = require('cho


利用 Cloudflare 邮件路由实现无限子邮箱配置指南
墨风如雪2026/3/24

上几期文章我介绍了怎么把域名托管到CLoudFlare和免费白嫖CF CDN的操作,这次我演示的是我日常最喜欢的功能之一,邮箱路由功能。可以只需要一个域名就可以拥有属于自己的邮箱,而且可以创建无限的子邮箱提供使用。 在这里你不需要搭建复杂的邮局,只要你有一个托管在 Cloudflare 的域名,就可以用任意的前缀邮箱来注册你想要的账号,所有邮件都会自动转发到你指定的主邮箱里面,接收验证码会非常的方便。 强大的开源资源库 在正式配置之前,我这里先介绍一个收集了十分多使用CLoudFlare免费资源


告别登录中断:前端双 Token无感刷新
发现一只大呆瓜2026/3/16

前言 在前后端分离的项目中,为了安全,Token 通常会设置有效期。但如果 Token 过期时强制用户重新登录,会极大地破坏用户体验。如何做到在用户毫无察觉的情况下,自动完成 Token 的续期?本文将深度拆解 “双 Token 无感刷新” 的实现机制。 一、 为什么需要“无感刷新”? 举个简单例子,你正在某 App 编辑内容,中途切出几分钟,再切回来时,直接弹出登录页,提示“登录已过期,请重新登录”,这种场景很容易让用户流失。 传统的单 Token 方案存在一个两难境地: 有效期过短:用户操


构建无障碍组件之Switch Pattern
anOnion2026/3/8

Switch Pattern 详解:构建无障碍开关组件 开关(Switch)是一种模拟物理开关的控件,用于在两个状态(通常是"开"和"关")之间切换。在一些 UI 组件库中,它也被称为 Toggle(切换开关)。本文基于 W3C WAI-ARIA Switch Pattern 规范,详解如何构建无障碍的开关组件。 一、Switch 的定义与核心概念 1.1 什么是 Switch Switch 是一种特殊的二元状态控件,它: 模拟物理开关的行为 在两个互斥状态之间切换(开/关、启用/禁用) 与


纯 CSS 实现弹性文字效果
掘金安东尼2026/2/28

原文:How to Create a CSS-only Elastic Text Effect 翻译:TUARAN 欢迎关注 前端周刊,每周更新国外论坛的前端热门文章,紧跟时事,掌握前端技术动态。 每个字母单独动画的文字效果总是很酷、很吸睛。这类错峰动画通常依赖 JavaScript 库实现,对我们要实现的这种相对轻量的设计效果来说,代码往往偏重。本文将探索只用 CSS、无需 JavaScript 实现 fancy 文字效果的技巧(意味着需要手动拆分字符)。 截至撰写时,仅 Chrome 和

首页编辑器站点地图

本站内容在 CC BY-SA 4.0 协议下发布

Copyright © 2026 聚合阅读