Linux Shell 脚本编程之前一直没有系统的去学习,在写 Shell 脚本的时候总需要现查各种语法。本文章以编程语言的维度去系统的学习 Shell 脚本编程。
Shell
Linux Shell 是与 Linux 系统交互的一个应用程序,我们通过这个程序可以操作 Linux 系统的内核服务。
执行 $cat /etc/shells
,可以看到系统中现在可用的 Shell 解释器
# List of acceptable shells for chpass(1). # Ftpd will not allow users to connect who are not using # one of these shells. /bin/bash /bin/csh /bin/ksh /bin/sh /bin/tcsh /bin/zsh /usr/local/bin/zsh
现代的 Linux 系统中 /bin/sh
已经被 /bin/bash
,作为 Linux 默认的 Shell.
输入 $echo $SHELL
可以看到当前系统的 Shell.
Shell 脚本
Shell 脚本 (Shell Script),是为 Shell 编写的一个脚本程序。我们说的 Shell 通常指的是 Shell 脚本。
Shell 变量
Shell 脚本属于弱类型的脚本语言,在使用的时候不需要提前定义变量类型。
@H_502_30@可能的坑:
- 赋值变量不能有美元符号 (
$
) - 赋值语句等号 (
=
) 左右都不能有空格
变量定义
#!/bin/bash # 直接赋值 name="cizel" # 语句赋值 for file in `ls /etc`
变量使用
#!/bin/bash # 定义变量 name name="cizel" # 使用美元 ($) 符号 echo $name # 使用美元 ($) 符号和括号结合,常用于字符串拼接 echo ${name}
or
#!/bin/bash # 高级用法 # 默认值:如果变量没有声明,使用默认值 ${var=DEFAULT} echo ${name="ok"} # output: ok # 默认值:如果变量没有声明,或者为空字符串,使用默认值 ${var:=DEFAULT} name="" echo ${name:="ok"} # output: ok
Shell 数字运算
Shell 中的数字运算可以采用 $((num1 + num2))
的方式,例如:
可能的坑:
- Shell 中的变量默认是字符串,使用
result=1+2;echo $result
,输出会是1+2
- 数值运算的两个变量必须是
数字
或者数字字符串
,不然会报错
#!/bin/bash a=2 b="3" echo (($a + $b)) # output: 5 echo (($a - $b)) # output: -1 echo (($a * $b)) # output: 6 echo (($a / $b)) # output: 0 # 取模 / 求余 echo $(($a % $b)) # output: 1 # 乘方 echo $(($a ** $b)) # output: 8 # 复杂运算 echo $(($a + ($a * $b))) # output: 8
Shell 字符串
Shell 的字符串与 PHP 的字符串相同,分为 单引号字符串
和 双引号字符串
字符串定义
#!/bin/bash $name="cizel" #单引号中变量和符号不会被解析 echo 'my name is ${name}' # output: my name is ${name} #双引号中变量和符号会被解析 echo 'my name is ${name}' # output: my name is $shizhen
字符串连接
#!/bin/bash name="cizel" echo $name $name # output:cizel cizel
字符串长度
#!/bin/bash name="cizel" echo ${#name} # output: 5
字符串截取
#!/bin/bash name="my name is cizel" echo ${name:2} # output: name is cizel echo ${name:2:5} # output: name
字符串删除
${变量名#substring 正则表达式} 从字符串 开头 开始配备 substring
,删除匹配上的表达式。
${变量名 %substring 正则表达式} 从字符串 结尾 开始配备 substring
,删除匹配上的表达式。
#!/bin/bash test="/home/work/.vimrc" echo ${test#/home} # output: /work/.vimrc
or
#!/bin/bash # 高级用法 test="/home/work/.vimrc" # 快速获取文件名 echo ${test##*/} # output: .vimrc # 快速获取路径 echo ${test%/*} # output: /home/work
字符串替换
使用内置的字符串替换,会比 awk
,sed
,expr
的性能更好,
${变量 / 查找 / 替换值} 一个"/"表示替换第一个,"//"表示替换所有。
#!/bin/bash test="/home/work/.vimrc" echo ${test/.vimrc/.zshrc} # output: /home/work/.zshrc echo ${test/w*k/cizel} # output: /home/cizel/.vimrc
Shell 逻辑运算
在 Shell 中,使用 test
来进行逻辑判断。与其他编程语言有许多不同,如果为真返回 0
,假返回 1
.
可能的坑:
- 逻辑判断结果真返回
0
,假返回1
- 使用
-gt
,-lt
,-ge
,-le
,-ne
,-eq
替换>
,<
,>=
,<=
,!=
,=
做数值比较 - 与或非运算符使用
-a
,-o
,!
替换&
|
!
数值比较
数值比较的运算符和汇编语言中类似,常见的 5 种数值比较如下:
符号 | 英文解释 | 中文解释 |
---|---|---|
-gt |
greater than | 大于 |
-lt |
less than | 小于 |
-ge |
greater equal | 大于等于 |
-le |
less equal | 小于等于 |
-ne |
not equal | 不等于 |
-eq |
equal | 等于 |
#!/bin/bash # 大于 test 3 -gt 2; echo $? # output: 0 # 小于 test 3 -lt 2; echo $? # output: 1 # 大于等于 test 3 -ge 2; echo $? # output: 0 # 小于等于 test 3 -le 2; echo $? # output: 1 # 不等于 test 3 -ne 2; echo $? # output: 0 # 等于 test 3 -eq 2; echo $? # output: 1
字符串比较
字符串比较的运算符如下表:
符号 | 解释 |
---|---|
= |
字符串等于 |
!= |
字符串不等 |
-z |
判断字符串长度是否为零 |
-n |
判断字符串长度是否大于零 |
#!/bin/bash # 字符串等于 test "my name is cizel" = "my name is cizel"; echo $? # output: 0 # 字符串不等 test "my name is cizel" = "my name is cz"; echo $? # output: 1 # 字符串长度判断 test -z "my name is cizel"; echo $? # output: 1 test -n "my name is cizel"; echo $? # output: 0
文件比较
符号 | 解释 |
---|---|
-e |
判断文件是否存在. |
-d |
判断文件是否为目录. |
-f |
判断文件是否为常规文件. |
-L |
判断文件是否为符号链接. |
-r |
判断文件是否可读. |
-w |
判断文件是否可写. |
-x |
判断文件是否可执行. |
#!/bin/bash ls -l # 当前目录有如下文件,lib 文件夹,run.sh 文件,sh 符号链接,当前角色:work # drwxr-xr-x 1 work work 4096 Jun 28 2018 lib # -rwxr-xr-x 1 work work 2364 Jul 7 2018 run.sh # lrwxrwxrwx 1 root root 4 May 26 2014 sh -> bash # 判断文件是否存在 test -e run.sh; echo $? # output: 0 # 判断目录是否存在 test -d lib; echo $? # output: 0 # 判断文件是否为常规文件 test -f run.sh; echo $? # output: 0 # 判断文件是否为符号链接 test -L sh; echo $? # output: 0 # 判断文件是否为符号链接 test -L sh; echo $? # output: 0 # 判断文件是否可读 / 写 / 执行 (当前角色 work,权限 rwx,可读可写可执行) test -r run.sh; echo $? # output: 0 test -w run.sh; echo $? # output: 0 test -x run.sh; echo $? # output: 0
逻辑连接
与其他编程语言一样,Shell 中也有与或非运算符。用于连接逻辑判断条件,形成复合的逻辑判断。
符号 | 英文解释 | 中文解释 |
---|---|---|
-a |
and | 与 |
-o |
or | 或 |
! |
-- | 非 |
#!/bin/bash # 与 test "1" = "1" -a "1" = "2"; echo $? # output: 1 # 或 test "1" = "1" -o "1" = "2"; echo $? # output: 0 #非 test ! "1" = "2"; echo $? # output: 0
Shell 选择结构
Shell 中的选择语句和其他编程语言类似,支持 if,if-else,if-elif,if-elif-else,case-esac 常见的条件选择方式
@H_502_30@可能的坑:
- if 条件的左括号 (
[
) 后必须有一个空格,右括号前 (]
) 必须有一个空格。if [空格expression空格] - if,elif 后面都需要加
then
然后添加语句
if 选择
#!/bin/bash var=`uname -s` if [ $var = "Linux" ]; then echo "Linux System" fi
if-else 选择
#!/bin/bash var=`uname -s` if [ $var = "Linux" ]; then echo "Linux System" else echo "Other System" fi
if-elif 选择
#!/bin/bash var=`uname -s` if [ $var = "Linux" ]; then echo "Linux System" elif [ $var = "FreeBSD" ]; then echo "FreeBSD System" fi
if-elif-else 选择
#!/bin/bash var=`uname -s` if [ $var = "Linux" ]; then echo "Linux System" elif [ $var = "FreeBSD" ]; then echo "FreeBSD System" else echo "Other System" fi
case-esac 选择
case-esac 与常用的 switch-case 类似,可以对比 if-elif-else 选择食用
#!/bin/bash var=`uname -s` case $var in "Linux") echo "Linux System" ;; "FreeBSD") echo "FreeBSD System" ;; *) echo "Other System" ;; esac
Shell 循环结构
for 循环
常见的类似 c 语言的写法
#!/bin/bash # 打印 1-10,必须使用双括号,使符号转移 for ((i=1; i<=10; i++)); do echo $i done
in 的方法 (常用)
#!/bin/bash for i in {1..10}; do echo $i done
while 循环
#!/bin/bash count=1 while [ $count -lt 3 ]; do echo $count count=$((count + 1)) done
until 循环
直到条件为真时,停止循环
#!/bin/bash count=1 until [ $count -eq 3 ]; do echo $count count=$((count + 1)) done
Shell 函数
Shell 函数,使用 $1..$n 的方式接收参数
#!/bin/bash my_func() { echo "my function" echo "params 1: $1" echo "params 2: $2" echo "params 3: $3" } my_func 1 2 3
Shell 加载脚本
Shell 中使用 source
命令可以加载其他文件到当前 Shell 脚本中
# echo.sh echo() { command printf %s\\n "$*" 2>/dev/null }
#!/bin/bash source echo.sh echo 123