标签:修改 pre 调试 bre span who 类型 bash art
一、概述
Shell是一种具备特殊功能的程序,它提供了用户与内核进行交互操作的一种接口。它接收用户输入的命令,并把它送入内核去执行。内核是Linux系统的心脏,从开机自检就驻留在计算机的内存中,直到计算机关闭为止,而用户的应用程序存储在计算机的硬盘上,仅当需要时才被调入内存。Shell是一种应用程序,当用户登录Linux系统时,Shell就会被调入内存去执行。Shell独立于内核,它是连接内核和应用程序的桥梁,并由输入设备读取命令,再将其转为计算机可以理解的机械码,Linux内核才能执行该命令。
二、优势
Shell脚本语言的好处是简单、易学、易用,适合处理文件和目录之类的对象,以简单的方式快速完成某些复杂的事情通常是创建脚本的重要原则,脚本语言的特性可以总结为以下几个方面:
1、语法和结构通常比较简单。
2、学习和使用通常比较简单,
3、通常以容易修改程序的“解释”作为运行方式,而不需要“编译。
4、程序的开发产能优于运行效能。
5、Shell脚本语言是Linux/Unix系统上一种重要的脚本语言,在Linux/Unix领域应用极为广泛,熟练掌握Shell脚本语言是一个优秀的Linux/Unix开发者和系统管理员必经之路。利用Shell脚本语言可以简洁地实现复杂的操作,而且Shell脚本程序往往可以在不同版本的Linux/Unix系统上通用。
三、变量
1、用户自定义变量
这种变量只支持字符串类型,不支持其他字符,浮点等类型,常见有这 3 个前缀:
a、unset:删除变量
b、readonly:标记只读变量
c、export:指定全局变量
#!/bin/bash
echo "定义普通变量"
CITY=CHENGDU
echo "定义全局变量"
export NAME=cdeveloper
echo "定义只读变量"
readonly AGE=21
echo "打印变量的值"
echo $CITY
echo $NAME
echo $AGE
echo "删除 CITY 变量"
unset CITY
# 不会输出 CHENGDU
echo $CITY
运行结果:
定义普通变量
定义全局变量
定义只读变量
打印变量的值
CHENGDU
cdeveloper
21
删除 CITY 变量
2、预定义变量
预定义变量常用来获取命令行的输入,有下面这些:
a、$0 :脚本文件名
b、$1-9 :第 1-9 个命令行参数名
c、$# :命令行参数个数
d、$@ :所有命令行参数
e、$* :所有命令行参数
f、$? :前一个命令的退出状态,可用于获取函数返回值
g、$$ :执行的进程 ID
编辑脚本1.sh
#!/bin/bash
echo "print $"
echo "1、打印脚本文件名$0"
echo -e "\$0 = $0 \n"
echo "2、打印第一个命令行参数$1"
echo -e "\$1 = $1\n"
echo "3、打印第二个命令行参数$2"
echo -e "\$2 = $2\n"
echo "4、打印命令行参数个数$#"
echo -e "\$# = $#\n"
echo "5、打印所有命令行参数$@"
echo -e "\$@ = $@\n"
echo "6、打印所有命令行参数"
echo -e "\$* = $*\n"
echo "7、打印前一个命令的退出状态"
echo -e "\$? = $?\n"
echo "8、打印执行的进程ID"
echo "\$\$ = $$"
执行结果:
print $
1、打印脚本文件名1.sh
$0 = 1.sh
2、打印第一个命令行参数1
$1 = 1
3、打印第二个命令行参数2
$2 = 2
4、打印命令行参数个数4
$# = 4
5、打印所有命令行参数1 2 3 4
$@ = 1 2 3 4
6、打印所有命令行参数
$* = 1 2 3 4
7、打印前一个命令的退出状态
$? = 0
8、打印执行的进程ID
$$ = 30839
3、环境变量
环境变量默认就存在,常用的有下面这几个:
a、HOME:用户主目录
b、PATH:系统环境变量 PATH
c、TERM:当前终端
d、UID:当前用户 ID
e、PWD:当前工作目录,绝对路径
编辑1.sh
#!/bin/bash
echo -e "print env\n"
echo "家目录"
echo -e "\$HOME=$HOME\n"
echo "环境变量"
echo -e "\$PATH=$PATH\n"
echo "当前终端"
echo -e "\$TERM=$TERM\n"
echo "当前工作目录"
echo -e "\$PWD=$PWD\n"
echo "当前用户ID"
echo "\$UID=$UID"
执行结果
print env
家目录
$HOME=/root
环境变量
$PATH=/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/application/jdk1.8.0_181/jre/bin:/application/jdk1.8.0_181/bin:/usr/local/rsync/bin:/application/sersync/bin:/root/bin
当前终端
$TERM=xterm
当前工作目录
$PWD=/script
当前用户ID
$UID=0
四、shell 运算
1、我们经常需要在 Shell 脚本中计算,掌握基本的运算方法很有必要,下面就是 4 种比较常见的运算方法,功能都是将 m + 1:
a、m=$[ m + 1 ]
b、m=expr $m + 1 # 用 “ 字符包起来
c、let m=m+1
d、m=$(( m + 1 ))
e、实例,编辑1.sh
#!/bin/bash
m=1
m=$[ m + 1 ]
echo $m
m=`expr $m + 1`
echo $m
# 注意:+ 号左右不要加空格
let m=m+1
echo $m
m=$(( m + 1 ))
echo $m
执行结果:
2
3
4
5
2、文件测试
a、-d:测试是否为目录(Directory)
b、-e:测试目录或文件是否存在(Exist)
c、-f:测试是否为文件(File)
d、-r:测试当前用户是否有权限读取(Read)
e、-w:测试当前用户是否有权限写入(Write)
f、-x:测试当前用户是否有权限执行(eXcute)
g、实例
[root@wohaoshuai1 script]# [ -d /media/cdrom ] && echo "YES" || echo "no"
no
3、数值比较
a、-eq:等于
b、-ne:不等于
c、-gt:大于
d、-lt:小于
e、-le:小于等于
f、-ge:大于或等于
g、实例 编辑1.sh
#!/bin/bash
[ $(who | wc -l) -eq 2 ] && echo "等于2"
[ $(who | wc -l) -ne 2 ] && echo "不等于2"
[ $(who | wc -l) -gt 2 ] && echo "大于2"
[ $(who | wc -l) -lt 2 ] && echo "小于2"
[ $(who | wc -l) -ge 2 ] && echo "大于或等于2"
[ $(who | wc -l) -le 2 ] && echo "小于或等于2"
4、字符串比较
a、= :字符串内容相同
b、!=:字符串内容不同,!表示相反的意思
c、-z:字符串内容为空
d、-n:字符串内容不为空
[root@wohaoshuai1 script]# echo $a
1
[root@wohaoshuai1 script]# [ $a = "b" ] && echo "a等于b" || echo "a不等于b"
a不等于b
[root@wohaoshuai1 script]# [ $a != "b" ] && echo "a不等于b" || echo "a等于b"
a不等于b
[root@wohaoshuai1 script]# [ -z $a ] && echo "a为空" || echo "a不为空"
a不为空
[root@wohaoshuai1 script]# [ -n $a ] && echo "a不为空" || echo "a为空"
a不为空
5、逻辑测试
a、-a 或 && : 逻辑与,“而且” 的意思
b、-o 或 || : 逻辑或,“或者” 的意思
c、! :逻辑否
6、[] 与 [[]] 区别
a、[[]] 用 "&&" 而不是"-a" 表示逻辑与,用"||"而不是"-o"表示逻辑或。
[root@wohaoshuai1 ~]# [[ 1 < 2 && b > a ]] && echo "true" || echo "false"
true
[root@wohaoshuai1 ~]# [[ 1 < 2 -a b > a ]] && echo "true" || echo "false"
-bash: syntax error in conditional expression
-bash: syntax error near `-a‘
b、[ ... ]为shell命令,所以在其中的表达式应是它的命令行参数,所以串比较操作符”>” 与”<”必须转义,否则就变成IO改向操作符了。[[中"<"与">"不需转义:
[root@wohaoshuai1 ~]# [ 1 \< 2 -a b \> a ] && echo "true" || echo "false"
true
c、[[ ... ]]进行算术扩展,而[ ... ]不做
[root@wohaoshuai1 ~]# [[ 99+1 -eq 100 ]] && echo "true" || echo "false"
true
[root@wohaoshuai1 ~]# [ 99+1 -eq 100 ] && echo "true" || echo "false"
-bash: [: 99+1: integer expression expected
false
[root@wohaoshuai1 ~]# [ $((99+1)) -eq 100 ] && echo "true" || echo "false"
true
d、[[]]能用正则,而[]不行,并且[[]]在正则匹配中不能使用双引号。
[root@wohaoshuai1 ~]# [[ "test.php" = *.php ]] && echo "true" || echo "false"
true
[root@wohaoshuai1 ~]# [ "test.php" = *.php ] && echo "true" || echo "false"
false
[root@wohaoshuai1 ~]# [[ "test.php" = "*.php" ]] && echo "true" || echo "false"
false
五、shell 语句
1、if 语句
a、这个跟高级语言的 if - else - if
类似,只是格式有些不同而已。
#!/bin/bash
read -p "请输入整数:" VART
if [ $VART -eq 10 ];then
echo "true"
elif [ $VART -gt 10 ];then
echo "大于10"
else
echo "false"
fi
执行结果:
[root@wohaoshuai1 script]# sh 1.sh
请输入整数:1
false
[root@wohaoshuai1 script]# sh 1.sh
请输入整数:10
true
[root@wohaoshuai1 script]# sh 1.sh
请输入整数:100
大于10
2、case语句,case 语句有些复杂,要注意格式
#!/bin/bash
read -p "请输入一个字符,并按enter结束:" key
case "$key" in
[./]|[a-z]|[A-Z])
echo "你输入的是字母"
;;
[0-9])
echo "你输入的是数字"
;;
*)
echo "你输入的是空格,功能键或其他控制字符"
esac
执行结果:
请输入一个字符,并按enter结束:a
你输入的是字母
[root@wohaoshuai1 script]# sh 1.sh
请输入一个字符,并按enter结束:.
你输入的是字母
[root@wohaoshuai1 script]# sh 1.sh
请输入一个字符,并按enter结束:/
你输入的是字母
3、for循环
#!/bin/bash
# 普通 for 循环
echo "普通for循环如下:"
for ((i = 1; i <= 3; i++))
do
echo $i
done
echo -e "\n"
# VAR 依次代表每个元素
echo "第二种for循环:"
for VAR in `seq 1 10`
do
echo $VAR
done
执行结果:
普通for循环如下:
1
2
3
第二种for循环:
1
2
3
4
5
6
7
8
9
10
4、while循环
#!/bin/bash
VAR=1
# 如果 VAR 小于 10,就打印出来
while [ $VAR -lt 10 ]
do
echo $VAR
# VAR 自增 1
VAR=$[ $VAR + 1 ]
done
执行结果:
1
2
3
4
5
6
7
8
9
5、until循环,until 语句与while循环的不同点是它的结束条件为 1
#!/bin/bash
i=0
# i 大于 5 时,循环结束
until [[ "$i" -gt 5 ]]
do
echo $i
i=$[ $i + 1 ]
done
运行结果:
0
1
2
3
4
5
6、break,Shell 中的 break
用法与高级语言相同,都是跳出循环。
#!/bin/bash
for VAR in `seq 1 3`
do
# 如何 VAR 等于 2 就跳出循环
if [ $VAR -eq 2 ];then
break
fi
echo $VAR
done
执行结果:
[root@wohaoshuai1 script]# sh 1.sh
1
7、continue,用来跳过本次循环,进入下一次循环
#!/bin/bash
for VAR in 1 2 3
do
# 如果 VAR 等于 2,就跳过,直接进入下一次 VAR = 3 的循环
if [ $VAR -eq 2 ];then
continue
fi
echo $VAR
done
执行结果:
[root@wohaoshuai1 script]# sh 1.sh
1
3
六、shell函数
1、定义函数,有两种常见的格式
#!/bin/bash
function hello_world()
{
echo "hello world fun"
echo $1 $2
return 1
}
hello()
{
echo "hello fun"
}
2、调用函数
#!/bin/bash
function hello_world()
{
echo "hello world fun"
echo $1 $2
return 1
}
hello()
{
echo "hello fun"
}
# 1. 直接用函数名调用 hello 函数
echo "1. 直接用函数名调用 hello 函数执行结果:"
hello
echo -e "\n"
# 2. 使用「函数名 函数参数」来传递参数
echo "2. 使用「函数名 函数参数」来传递参数执行结果:"
hello_world 1 2
echo -e "\n"
# 3. 使用「FUN=`函数名 函数参数`」 来间接调用
echo "3. 使用「FUN=‘函数名 函数参数‘」 来间接调用:"
FUN=`hello_world 1 2`
echo -e $FUN
执行结果:
[root@wohaoshuai1 script]# sh 1.sh
1. 直接用函数名调用 hello 函数执行结果:
hello fun
2. 使用「函数名 函数参数」来传递参数执行结果:
hello world fun
1 2
3. 使用「FUN=‘函数名 函数参数‘」 来间接调用:
hello world fun 1 2
3、获取返回值,使用 $?获取返回值
#!/bin/bash
function hello_world()
{
echo "hello world fun"
echo $1 $2
return 2
}
hello()
{
echo "hello fun"
}
# 1. 直接用函数名调用 hello 函数
echo "1. 直接用函数名调用 hello 函数执行结果:"
hello
echo -e "\n"
# 2. 使用「函数名 函数参数」来传递参数
echo "2. 使用「函数名 函数参数」来传递参数执行结果:"
hello_world 1 2
echo "返回值为:$?"
echo -e "\n"
# 3. 使用「FUN=`函数名 函数参数`」 来间接调用
echo "3. 使用「FUN=‘函数名 函数参数‘」 来间接调用:"
FUN=`hello_world 1 2`
echo -e $FUN
执行结果:
[root@wohaoshuai1 script]# sh 1.sh
1. 直接用函数名调用 hello 函数执行结果:
hello fun
2. 使用「函数名 函数参数」来传递参数执行结果:
hello world fun
1 2
返回值为:2
3. 使用「FUN=‘函数名 函数参数‘」 来间接调用:
hello world fun 1 2
4、定义本地变量
fun()
{
local x=1
echo $x
}
七、Shell 调试
1、使用下面的命令来检查是否有语法错误
sh -n 1.sh
2、使用下面的命令来执行并调试 Shell 脚本
sh -x 1.sh
3、实例
#!/bin/bash
for VAR in 1 2 3
do
if [ $VAR -eq 2 ];then
continue
fi
echo $VAR
done
执行结果:
[root@wohaoshuai1 script]# sh -x 1.sh
+ for VAR in 1 2 3
+ ‘[‘ 1 -eq 2 ‘]‘
+ echo 1
1
+ for VAR in 1 2 3
+ ‘[‘ 2 -eq 2 ‘]‘
+ continue
+ for VAR in 1 2 3
+ ‘[‘ 3 -eq 2 ‘]‘
+ echo 3
3
其中带有 +
表示的是 Shell
调试器的输出,不带 +
表示我们程序的输出。
八、Shell数组
1、数组分为两种类型,一是数值类型,二是字符串类型。
a、数值类型的数组:一对括号表示数组,数组中元素之间使用“空格”来隔开。
arr_number=(1 2 3 4 5)
b、字符串类型数组,同样,使用一对括号表示数组,其中数组中的元素使用双引号或者单引号包含,同样使用“空格”来隔开。
arr_string=("abc" "edf" "sss"); 或者 arr_string=(‘abc‘ ‘edf‘ ‘sss‘);
2、数组的操作
a、获取数组长度,arr_length=${#arr_number[*]}或${#arr_number[@]}均可,即形式:${#数组名[@/*]} 可得到数组的长度。
[root@wohaoshuai1 script]# a=(1 2 3 4 5)
[root@wohaoshuai1 script]# echo "${#a[*]}"
5
[root@wohaoshuai1 script]# echo "${#a[@]}"
5
b、读取某个下标的值,arr_index2=${arr_number[2]},即形式:${数组名[下标]}
[root@wohaoshuai1 script]# a=(1 2 3 4 5)
[root@wohaoshuai1 script]# echo "${a[2]}"
3
c、对某个下标赋值。
1)、若下标元素已经存在那么会修改该下标的值为新的指定值。
[root@wohaoshuai1 script]# a=(1 2 3 4 5)
[root@wohaoshuai1 script]# a[2]=100
[root@wohaoshuai1 script]# echo ${a[*]}
1 2 100 4 5
[root@wohaoshuai1 script]# echo ${a[@]}
1 2 100 4 5
2)、如果指定的下标已经超过当前数组的大小,如上述的a的大小为5,指定下标为10或者11或者大于5的任意值那么新赋的值会被追加到数组的尾部。
[root@wohaoshuai1 script]# a=(1 2 3 4 5)
[root@wohaoshuai1 script]# a[2]=100
[root@wohaoshuai1 script]# echo ${a[*]}
1 2 100 4 5
[root@wohaoshuai1 script]# echo ${a[@]}
1 2 100 4 5
[root@wohaoshuai1 script]# a[10]=200
[root@wohaoshuai1 script]# echo ${a[@]}
1 2 100 4 5 200
[root@wohaoshuai1 script]# echo "${a[10]}"
200
[root@wohaoshuai1 script]# echo "${a[5]}"
d、删除操作
1)、清空某个元素
[root@wohaoshuai1 script]# a=(1 2 3 4 5)
[root@wohaoshuai1 script]# unset a[1]
[root@wohaoshuai1 script]# echo "${a[*]}"
1 3 4 5
2)、清空整个数组
[root@wohaoshuai1 script]# a=(1 2 3 4 5)
[root@wohaoshuai1 script]# unset a
[root@wohaoshuai1 script]# echo "${a[*]}"
e、分片访问,分片访问形式为:${数组名[@或*]:开始下标:结束下标},注意,不包括结束下标元素的值。
1)、从索引2开始访问,访问4个,如果个数不够默认会访问到最后一个
[root@wohaoshuai1 script]# a=(1 2 3 4 5)
[root@wohaoshuai1 script]# echo ${a[@]:2:4}
3 4 5
f、模式替换,形式为:${数组名[@或*]/模式/新值}
[root@wohaoshuai1 script]# a=("aa","bb","cc","aa")
[root@wohaoshuai1 script]# echo ${a[@]/aa/x}
x,bb,cc,aa
[root@wohaoshuai1 script]# echo ${a[@]//aa/x}
x,bb,cc,x
[root@wohaoshuai1 script]# echo ${a[@]}
aa,bb,cc,aa
g、数组的遍历
#!/bin/bash
b=(aa bb cc dd ee)
for i in ${b[@]}
do
echo ${i}
done
执行结果:
[root@wohaoshuai1 script]# sh 1.sh
aa
bb
cc
dd
ee
九、Shell字符串处理
1、获取字符串长度
[root@wohaoshuai1 script]# str="abcdefg"
[root@wohaoshuai1 script]# echo "${#str}"
7
2、对字符串进行大小写转换
a、将变量值中的小写字母转换为大写
[root@wohaoshuai1 script]# echo "${str}"
abcdefg
[root@wohaoshuai1 script]# echo "${str^^}"
ABCDEFG
b、将变量中的大写字母转换为小写
[root@wohaoshuai1 script]# echo "${str}"
ABCDEFG
[root@wohaoshuai1 script]# echo "${str,,}"
abcdefg
3、当变量值为空或非空时操作变量
a、${var:=value},意思为如果var为空则返回value,并将value赋给var,如果var不为空则返回var本身的值,var不为空时,var值不会被改变,var为空时,var的值会被设置成指定值。
#!/bin/bash
a="wohaoshuai"
b=${a:="123"}
echo -e "b为:${b}\n"
c=${d:="456"}
echo -e "c为:${c}\n"
echo -e "d为:${d}\n"
执行结果:
[root@wohaoshuai1 script]# sh 1.sh
b为:wohaoshuai
c为:456
d为:456
b、${var:-value},意思为如果var为空,则返回value,如果var不为空,则返回var的值,无论var是否为空,var本身的值不会改变。
#!/bin/bash
a="wohaoshuai"
b=${a:-"123"}
echo -e "b为:${b}\n"
c=${d:-"456"}
echo -e "c为:${c}\n"
echo -e "d为:${d}\n"
执行结果:
[root@wohaoshuai1 script]# sh 1.sh
b为:wohaoshuai
c为:456
d为:
c、${var:+value},意思为如果var不为空,则返回value,如果var为空,则返回空值,无论var是否为空var本身的值不会改变。
#!/bin/bash
a="wohaoshuai"
b=${a:+"123"}
echo -e "b为:${b}\n"
c=${d:+"456"}
echo -e "c为:${c}\n"
echo -e "d为:${d}\n"
执行结果为:
[root@wohaoshuai1 script]# sh 1.sh
b为:123
c为:
d为:
d、${var:?error_info},意思为如果var为空,那么在当前终端打印error_info,如果var的值不为空,则返回var的值,无论var是否为空,var本身的值都不会改变。
#!/bin/bash
a="wohaoshuai"
b=${a:?"123"}
echo -e "b为:${b}\n"
c=${d:? "err"}
echo -e "c为:${c}\n"
echo -e "d为:${d}\n"
执行结果:
[root@wohaoshuai1 script]# sh 1.sh
b为:wohaoshuai
1.sh: line 7: d: err
4、从指定位置截取字符串,截取到字符串的末尾。
a、从正数第四个字符以后开始截取,直到字符串的末尾(从0开始计数)
[root@wohaoshuai1 script]# a="abcdefg"
[root@wohaoshuai1 script]# echo ${a:4}
efg
b、从倒数第四个字符开始截取,直到字符串的末尾。并且冒号与负号之间必须存在任意字符(通常使用0或空格占位,当然用其它字符也可以),否则无法起到截取字符串的作用。
[root@wohaoshuai1 script]# echo ${a:0-4}
defg
[root@wohaoshuai1 script]# echo ${a: -4}
defg
[root@wohaoshuai1 script]# echo ${a:b-4}
defg
5、从指定位置截取字符串,并且截取指定的长度
a、从第四个字符以后开始截取,截取五个字符
[root@wohaoshuai1 script]# a="abcdefghijklmn"
[root@wohaoshuai1 script]# echo ${a:4:5}
efghi
b、从倒数第八个开始截取,截取5个字符
[root@wohaoshuai1 script]# echo ${a:0-8:5}
ghijk
[root@wohaoshuai1 script]# echo ${a:b-8:5}
ghijk
[root@wohaoshuai1 script]# echo ${a: -8:5}
ghijk
c、Centos7中截取长度可以为负数,Centos6中不行,Centos7从正数第四个字符以后开始截取到字符串末尾,再将截取后的字符串的最后3个字符删除。
[root@wohaoshuai1 script]# echo ${a:4}
efghijklmn
[root@wohaoshuai1 script]# echo ${a:4:-3}
efghijk
d、Centos7从倒数第四个字符开始截取,截取到字符串的末尾,再将截取后的字符串的最后一个字符删除。
[root@wohaoshuai1 script]# echo ${a:0-4}
klmn
[root@wohaoshuai1 script]# echo ${a:0-4:-1}
klm
6、替换字符串操作
a、将变量值中第一个遇到的str1替换成str2
[root@wohaoshuai1 script]# a="www"
[root@wohaoshuai1 script]# echo ${a/w/abc}
abcww
b、将变量值中的所有遇到的str1替换成str2
[root@wohaoshuai1 script]# echo ${a//w/abc}
abcabcabc
标签:修改 pre 调试 bre span who 类型 bash art
原文地址:https://www.cnblogs.com/luweiweicode/p/14256279.html