2

第一个shell脚本

#!/bin/bash
echo "hello, world"

运行程序可以作为解释器参数或者作为可执行程序

bash test.sh
chmod +x test.sh
test.sh

变量

命名

name="huruji"

需要注意的是变量名与等号之间不能有空格。

使用

echo $name
echo ${name}

使用在变量名前添加$即可,{}表示这个变量名的边界。

只读变量

name="huruji"
readonly name

使用readonly可以将变量定义为只读变量,只读变量不能再次赋值

删除变量

name="huruji"
unset name

使用unset删除变量,之后不能再使用。

字符串

name="huruji"

echo "my name is $name"

字符串可以使用单引号和双引号,单引号中不能包含单引号,即使转义单引号也不次那个,双引号则可以,双引号也可以使用字符串。

拼接

name="huruji"
hello="my name is ${name}"

获取字符串长度

str="huruji"
echo ${#str} #6

提取子字符串

str="huruji"
echo ${str:2:3}

从字符串的第二个字符开始提取3个字符,输出ruj

查找

str="huruji"
echo `expr index "$str" u`

此时输出2,因为此时第一个字符位置从1开始

数组

定义

names=("huruji" "greywind" "xie")
echo ${names[0]}
echo ${names[2]}

读取

echo ${names[2]}
echo ${names[@]}

如上例子,使用@可以获取数组中的所有元素

获取长度

length=${#names[@]}
length=${#names[*]}

Shell参数传递

执行Shell脚本的时候,可以向脚本传递参数,在Shell中获取这些参数的格式为$n,即$1,$2.......,

echo "第一个参数是:$1"
echo "第一个参数是:$2"
echo "第一个参数是:$3"

运行

chmod +x test.sh
test.sh 12 13 14

则此时输出:

第一个参数是:12
第一个参数是:13
第一个参数是:14

此外,还有其他几个特殊字符来处理参数

  • $#:传递脚本的参数个数
  • $*:显示所有的参数
  • $$:脚本当前运行的进程ID号
  • $!:后台运行的最后一个进程的ID号
  • $@:返回所有参数
  • $-:显示Shell的使用的当前选项
  • $?:退出的状态,0表示没有错误,其他则表示有错误

运算

算数运算
原生bash不支持简单的数学运算,可以借助于其他命令来完成,例如awk和expr,其中expr最常用。expr是一款表达式计算工具,使用它能完成表达式的求值操作。

val=`expr 2 + 2`
echo $val

需要注意的是运算符两边需要空格,且使用的是反引号。
算术运算符包括:+ - × / % = == !=

关系运算
关系运算只支持数字,不支持字符串,除非字符串的值是数字。

a=12
b=13
if [ $a -eq $b ]
    then
    echo "相等"
else
    echo "不等"
fi
  • -eq:是否相等
  • -ne:是否不等
  • -gt:大于
  • -lt:小于
  • -ge:大于等于
  • -le:小于等于

布尔运算

  • !:非
  • -o:或
  • -a:与

逻辑运算符

  • &&:逻辑与
  • ||:逻辑或

字符串运算符

  • =:相等 [ $a = $b ]
  • !=:不等 [ $a != $b ]
  • -z:字符串长度是否为0,为0返回true [ -z $a ]
  • -n:字符串长度是否为0,不为0返回true [ -n $a ]
  • str:字符串是否为空,不为空返回true [ $a ]

文件测试运算符
用于检测Unix文件的各种属性。

  • -b:检测文件是否为块设备文件 [ -b $file ]
  • -c:检测文件是否为字符设备文件 [ -c $file ]
  • -d:检测文件是否为目录 [ -d $file ]
  • -f:检测文件是否为普通文件 [ -f $file ]
  • -g:检测文件是否设置了SGID位 [ -g $file ]
  • -k:检测文件是否设置了粘着位 [ -k $file ]
  • -p:检测文件是否是有名管道 [ -p $file ]
  • -u:检测文件是否设置了SUID位 [ -u $file ]
  • -r:检测文件是否可读 [ -r $file ]
  • -w:检测文件是否可写 [ -w $file ]
  • -x:检测文件是否可执行 [ -x $file ]
  • -s:检测文件大小是否大于0 [ -s $file ]
  • -e:检测文件是否存在 [ -e $file ]
file="/home/greywind/Desktop/learnShell/test.sh"

if [ -e $file ]
    then
    echo "文件存在"
else
    echo "文件不存在"
fi

if [ -r $file ]
    then
    echo "可读"
else
    echo "不可读"
fi

if [ -w $file ]
    then
    echo "可写"
else
    echo "不可写"
fi


if [ -x $file ]
    then
    echo "可执行"
else
    echo "不可执行"
fi

if [ -d $file ]
    then
    echo "是目录"
else
    echo "不是目录"
fi

if [ -f $file ]
    then
    echo "是普通文件"
else
    echo "不是普通文件"
fi

echo

echo在显示输出的时候可以省略双引号,使用read命令可以从标准输入中读取一行并赋值给变量

read name
echo your name is $name

换行使用转义n,不换行使用c
此外使用 > 可以将echo结果写入指定文件,这个文件不存在会自动创建

echo "it is a test" > "/home/greywind/Desktop/learnShell/hello"

使用反引号可以显示命令执行的结果,如date、history、pwd

echo `pwd`
echo `date`

printf

Shell中的输出命令printf类似于C语言中的printf(),
语法格式:

printf format-string [arguments...]
printf "%-10s %-8s %-4s\n" 姓名 性别 体重kg  
printf "%-10s %-8s %-4.2f\n" 郭靖 男 66.1234 
printf "%-10s %-8s %-4.2f\n" 杨过 男 48.6543 
printf "%-10s %-8s %-4.2f\n" 郭芙 女 47.9876 

test

test命令用于检查某个条件是否成立,可以进行数值、字符、文件三方面的测试

a=100
b=200
if test a == b
    then
    echo "相等"
else
    echo "不等"
fi

流程控制

if

a=100
b=200
if test $a -eq $b
    then
    echo "相等"
else
    echo "不等"
fi
a=100
b=200
if test $a -eq $b
    then
    echo "相等"
elif test $a -gt $b
    then
    echo "a大于b"
elif test $a -lt $b
    then
    echo "a小于b"
fi

for

for num in 1 2 3 4
do
    echo ${num}
done
num=10
for((i=1;i<10;i++));
do
    ((num=num+10))
done
echo $num

while

num=1
while [ $num -lt 100 ]
do
    ((num++))
done

echo $num

无限循环

while:
do
      command
done
while true
do
      command
done
for (( ; ; ))

until

until condition
do
      command
done

case

case 值 in
模式1)
    command1
    command2
    ...
    commandN
    ;;
模式2)
    command1
    command2
    ...
    commandN
    ;;
esac

需要注意的是与其他语言不同Shell使用;;表示break,另外没有一个匹配则使用*捕获该值

echo "输入1 2 3任意一个数字"
read num
case $num in
    1)echo "输入了1"
;;
    2)echo "输入了2"
;;
    3)echo "输入了3"
;;
    *)echo "输入的值不是1 2 3"
;;
esac

与其他语言类似,循环可以使用break和continue跳出

函数

函数定义
用户自定义函数可以使用或者不使用function关键字,同时指定了return值则返回这个值,如果没有return语句则以最后一条运行结果作为返回值。

function first(){
    echo "hello world"
}
first(){
    echo "hello world"
}

调用函数直接使用这个函数名即可

first

函数参数
调用函数可以传入参数,函数内部使用$n获取传入的参数,类似于运行程序使用时获取使用的参数,不过需要注意的是两位数以上应该使用{}告诉shell边界例如${12}、${20}

function add(){
    num=0;
    for((i=1;i<=$#;i++));
    do
        num=`expr $i + $num`
    done
    return $num
}
add 1 2 3 4 5
a=$?

echo $a

函数本身是一个命令,所以只能通过$?来获得这个返回值

输入输出重定向

在上文的例子中可以使用 > 可以将echo结果写入指定文件,这就是一种输出重定向,重定向主要有以下:

  • command > file:输出重定向至文件file
  • command < file:输入重定向至文件file
  • command >> file:输出以追加的方式重定向至文件file
  • n > file:将文件描述符为n的文件重定向至文件file
  • n >> file:将文件描述符为 n 的文件以追加的方式重定向到文件file
  • n >& m:将输出文件 m 和 n 合并
  • n <& m:将输入文件 m 和 n 合并
  • << tag:将开始标记 tag 和结束标记 tag 之间的内容作为输入

将whoami命令输出保存到user文件中

who > "./user"

使用cat命令就可以看到内容已经保存了,如果不想覆盖文件的内容那么就使用追加的方式即可。

who >> "./user"

Shell文件包含

Shell脚本可以包含外部脚本,可以很方便的封装一些公用的代码作为一个独立的文件,包含的语法格式如下:

. filename
# 或
source filename

如:
test1.sh

echo "hello world"

test.sh

source ./test1.sh

echo "hello"

灰风GreyWind
308 声望13 粉丝

灰风也叫忽如寄