掌握Shell脚本的10个核心技巧,涵盖变量安全、输入控制、条件判断、算术运算、字符串处理、参数传递、流程控制、错误处理与调试方法,强调语法细节如空格、引号、默认值处理及set -e的合理使用,提升脚本健壮性与可维护性。

Shell 脚本不是写一次就完事的工具,而是你每天和 Linux 打交道的“第二双手”。掌握基础语法不靠死记,而在于理解每个写法背后的逻辑和常见坑点。下面这 10 个技巧,覆盖变量、输入、判断、运算、错误处理等核心场景,全是新手上手时最常卡住的地方。
变量定义与安全取值
变量名不能有空格,等号两边也不能有空格;命名可用字母、数字、下划线,但不能以数字开头。更关键的是:未定义变量默认为空,容易导致命令异常(比如 rm -rf $DIR/* 中 DIR 为空会删当前目录)。
- 用 ${var:-default} 提供默认值,比如 name=${name:-"guest"}
- 用 ${var:?error} 强制报错并退出,适合关键参数缺失时提醒用户
- 局部变量加 local 前缀,避免函数内修改污染全局作用域
读取用户输入要带控制
read 看似简单,但没加限制容易让脚本卡住或收不到预期内容。
- 加 -p 显示提示,比如 read -p "请输入密码: " -s password(-s 不回显)
- 加 -t 5 设置超时,5 秒没输就继续执行后续逻辑
- 加 -r 防止反斜杠被转义,尤其读路径或命令时很关键
条件判断别只记符号,先看类型
Shell 的 [ ] 判断对数据类型很敏感:数字用 -eq、-gt,字符串用 = 或 !=,空值检查用 -z 或 -n。混用会静默失败。
- 判断文件是否存在:[ -f "$file" ]
- 判断字符串非空:[ -n "$str" ] 比 [ "$str" ] 更明确
- 数字比较务必加引号防空值报错:[ "$a" -gt "$b" ]
算术运算要选对方式
Shell 默认把所有变量当字符串,做数学得显式声明。三种常用方式各有所长:
- $((...)) 最推荐:支持变量、四则、括号优先级,比如 sum=$((a + b * 2))
- let 可读性稍差,适合简单赋值:let "c = a + b"
- expr 已逐渐淘汰,乘号需转义:expr $a \* $b
字符串截取不用 sed 和 awk
Shell 内置参数扩展就能快速切路径、取后缀、去前缀,又快又稳。
- 取目录:dir="${path%/*}"(从右往左删第一个 / 及右边)
- 取文件名:file="${path##*/}"(从左往右删最后一个 / 及左边)
- 去后缀:name="${filename%.*}";留后缀:ext="${filename##*.}"
传参处理要规范
脚本运行时的参数不是摆设,合理利用能提升复用性。
- $1、$2 是第 1、2 个参数;$# 是总数;$@ 安全展开所有参数(含空格)
- 用 shift 向左移位,适合解析多选项,比如处理 ./deploy.sh -e prod -v 2.1
- 配合 getopts 可自动解析短选项,比手工判断更健壮
流程控制注意缩进和空格
if、for、while 的结构看着松散,但空格和换行是语法的一部分。
- if [ condition ]; then 中的方括号是命令,前后必须有空格
- case 结尾用 ;;,不是 ;,漏掉会报错
- for 循环中 in 后面可直接跟列表,也可用 $(ls),但注意含空格文件名要用 while IFS= read -r 替代
错误处理不是加 set -e 就万事大吉
set -e 让任意命令失败就退出,看似保险,但有些命令天然返回非零(如 grep 找不到匹配),反而误杀脚本。
- 对可能失败的命令单独处理:command || echo "忽略此错误"
- 用 $? 检查上一条命令状态,再决定是否重试或跳过
- 临时关闭严格模式:set +e,处理完再 set -e
调试技巧比写代码还重要
刚写的脚本跑不通?别急着改逻辑,先看清它到底执行了什么。
- bash -x script.sh 直接打印每行执行过程
- 在脚本里加 set -x 开启,set +x 关闭,精准定位某一段
- 用 echo "DEBUG: var=$var" 插桩,比猜快十倍
脚本开头和权限别忽略
一个可执行脚本,两处细节决定它能不能跑起来。
- 第一行必须是 #!/bin/bash(或对应解释器路径),否则系统不知道用谁执行
- 保存后执行 chmod +x script.sh,否则 ./script.sh 会报 “Permission denied”
- 建议统一用 .sh 后缀,方便识别和编辑器语法高亮
基本上就这些。不复杂但容易忽略——真正卡住新手的,往往不是语法有多难,而是某个空格、某个引号、某个没赋值的变量。边写边验证,比一口气写完再调试高效得多。










