1、程序构成:
主要由指令和数据构成,数据可以是变量、文件、数组
2、数组介绍:
变量:存储单个元素的内存空间
数组:存储多个元素的连续的内存空间
3、数组的特点:
数组名:整个数组只有一个名字
数组索引:元素编号从“0”开始,数组名[索引]
注意:对于bash-4及之后的版本,支持自定义的索引格式,而不仅仅是0,1,2....等等数字格式。同时数组可以不事先声明,可以直接使用,但是如果不声明,它会把多个字符串的变量当做一个变量,因此使用数组前,应先声明数组。
4、声明数组:(有两种方式)
方式一:declare -a NAME //声明索引数组,用0,1,2,3....表示各元素
方式二:declare -A NAME //声明关联数组,自定义索引格式
5、给数组中元素赋值:
赋值方式 | 备注 |
①、一次只赋值一个元素: 格式:ARRAY_NAME[INDEX]=value 示例: [root@localhost ~]# declare -a animal [root@localhost ~]# animal[0]=pig [root@localhost ~]# animal[1]=sheep [root@localhost ~]# echo $animal //引用第“0”个元素 pig [root@localhost ~]# echo ${animal[1]} pig[1] [root@localhost ~]# |
|
②、一次赋值全部元素: 格式:ARRAY_NAME=("VALUE1" "VALUE2" "VALUE3" ...) //必须以空格键隔开各元素值 示例: [root@localhost ~]# weekdays=("monday" "tuesday" "wedsday") [root@localhost ~]# echo ${weekdays[2]} wedsday [root@localhost ~]# |
|
③、只赋值特定元素: 格式:ARRAY_NAME=([0]="VAL0" [1]="VAL1" [3]="VAL3") //可以不连续赋值,此种被称为稀疏数组。 示例: [root@localhost ~]# cigrate=([0]="najing" [1]="xiaosu" [4]="yuxi") [root@localhost ~]# echo ${cigrate[0]} najing [root@localhost ~]# echo ${cigrate[1]} xiaosu [root@localhost ~]# echo ${cigrate[2]} [root@localhost ~]# echo ${cigrate[3]} [root@localhost ~]# echo ${cigrate[4]} yuxi [root@localhost ~]# 注意:bash支持稀疏格式的数组。 |
|
④、read -a ARRAY_NAME //回车后输入输入元素 示例: [root@localhost ~]# read -a ARR_name jiangsu anhui hubei [root@localhost ~]# echo $ARR_name jiangsu [root@localhost ~]# |
|
⑤、关联数组: ARRAY_NAME[INDEX]=var //必须declare -A后才能使用关联数组 示例: [root@localhost ~]# declare -A world [root@localhost ~]# world[zh]=china [root@localhost ~]# world[en]=english [root@localhost ~]# echo $world [root@localhost ~]# echo $world[zh] [zh] [root@localhost ~]# echo $world[en] [en] [root@localhost ~]# |
6、数组元素引用:
格式: ${ARRAY_NAME[INDEX]} //必须加{ },因为不加,则只会引用ARRAY_NAME,而不是整个数组元素。
数组元素其他使用技巧:
①、数组元素的长度:(数组元素的个数) 格式一:${#ARRAY_NAME[*]} //如果不加*或@表示引用数组中的第一个元素的个数 格式二:${#ARRAY_NAME[@]} //如果不加*或@表示引用数组中的第一个元素的个数 示例: [root@localhost ~]# declare -a china //声明一个数组名为china [root@localhost ~]# china=("bejing" "hanghai" "guanzhou" "jiangsu") //给数组赋值 [root@localhost ~]# echo $china //此种将会输出数组中的第一个元素值 bejing [root@localhost ~]# echo ${#china[*]} //输出数组中元素的个数,即数组长度 4 [root@localhost ~]# echo ${#china[@]} //输出数组中元素的个数,即数组长度 4 [root@localhost ~]# echo ${#china} //若不加“*”或“@”,则表示输出数组中第一个元素字符的个数 6 [root@localhost ~]# |
|
②、数组中第一个元素的字符个数: 格式:${#ARRAY_NAME} 示例: [root@localhost ~]# declare -a china [root@localhost ~]# china=("bejing" "hanghai" "guanzhou" "jiangsu") [root@localhost ~]# echo ${#china} 6 |
|
③、引用或查看数组中的全部元素值 格式一:${ARRAY_NAME[*]} 格式二:${ARRAY_NAME[@]} [root@localhost ~]# declare -a china [root@localhost ~]# china=("bejing" "hanghai" "guanzhou" "jiangsu") [root@localhost ~]# echo ${chian[*]} bejing hanghai guanzhou jiangsu [root@localhost ~]# echo ${chian[@]} bejing hanghai guanzhou jiangsu [root@localhost ~]# |
|
④、数组元素切片 格式一:${ARRAY_NAME[@]:offset:number} 格式二:${ARRAY_NAME[*]:offset:number} offset:要略过的元素的个数,即偏移量,注意此处不是步长的概念。 number:要取出元素的个数,省略number时表示取偏移量之后的所有元素 示例: [root@localhost sh]# declare -a files [root@localhost sh]# files=(/etc/[Pp]*) [root@localhost sh]# echo ${files[@]} /etc/PackageKit /etc/pam.d /etc/pam_pkcs11 /etc/passwd /etc/passwd- /etc/pbm2ppa.conf /etc/pcp /etc/pcp.conf /etc/pcp.env /etc/Pegasus /etc/pinforc /etc/pkcs11 /etc/pki /etc/plymouth /etc/pm /etc/pnm2ppa.conf /etc/polkit-1 /etc/popt.d /etc/portreserve /etc/postfix /etc/ppp /etc/prelink.conf.d /etc/printcap /etc/profile /etc/profile.d /etc/protocols /etc/pulse /etc/purple /etc/python [root@localhost sh]# echo ${files[@]:2:2} /etc/pam_pkcs11 /etc/passwd [root@localhost sh]# echo ${files[*]:2:2} /etc/pam_pkcs11 /etc/passwd [root@localhost sh]# [root@localhost sh]# echo ${files[*]:10} /etc/pinforc /etc/pkcs11 /etc/pki /etc/plymouth /etc/pm /etc/pnm2ppa.conf /etc/polkit-1 /etc/popt.d /etc/portreserve /etc/postfix /etc/ppp /etc/prelink.conf.d /etc/printcap /etc/profile /etc/profile.d /etc/protocols /etc/pulse /etc/purple /etc/python [root@localhost sh]# |
|
⑤、向数组中追加元素(非稀疏格式) 格式一:ARRAY_NAME[{#ARRAY_NAME[@]}]=value //标红部分表示数组长度,即要追加元素的下标 格式二:ARRAY_NAME[{#ARRAY_NAME[*]}]=value //标红部分表示数组长度,即要追加元素的下标 |
|
⑥、删除数组中的某元素 撤销变量:#unset VAR //VAR是变量名 撤销数组中的某元素:#unset ARRAY[INDEX] //注意:不要加“$”因为撤销的是元素 |
7、数组练习示例:
①、生成10个随机数,并找出其中的最大值和最小值。
[root@localhost sh]# cat array_max_min.sh
#!/bin/bash
declare -i max=0
declare -i min=0
declare -a array_int
for i in `seq 1 10` ;do
array_int[$i]=$RANDOM
echo ${array_int[$i]}
if [ $i -eq 1 ]; then
max=${array_int[$i]}
min=${array_int[$i]}
elif [ ${array_int[$i]} -gt $max ];then
max=${array_int[$i]}
elif [ ${array_int[$i]} -lt $min ];then
min=${array_int[$i]}
fi
done
echo "max=$max ; min=$min"
[root@localhost sh]#
②、生成10个随机数字,而后进行由小到大进程排序。
③、写一个脚本实现如下功能
定义一个数组,数组中的元素是/var/log目录下所有以".log"结尾的文件,统计其下标为偶数的文件的行数之和。
[root@localhost sh]# cat array_pattern.sh
#!/bin/bash
declare -a files
files=(/var/log/*.log)
for i in $(seq 0 $[${#files[*]}-1]);do
if [ $[$i%2] -eq 0 ];then
let lines+=$(wc -l ${files[$i]} | cut -d" " -f1)
fi
done
echo "lines : $lines"
[root@localhost sh]#
8、bash内置字符串处理:
8.1字符串切片: 格式:${var:offset:number} //var是变量名,offset是偏移量,number是取出的个数,整个意思是:取字符串的子串, 示例: [root@localhost ~]# name=changcheng [root@localhost ~]# echo ${name} changcheng [root@localhost ~]# echo ${name:2} //取偏移量之后所有字符 angcheng [root@localhost ~]# echo ${name:2:3} //取偏移量之后的3个字符 ang [root@localhost ~]# |
8.2、基于模式取子串: 格式一:${var#*word} //其中word是指定的分隔符,整个表达式的功能是:自左向右查找var变量所存储的字符串中,第一次出现的word分隔符,删除字符串开头至此分隔符之间的所有字符。 示例: [root@localhost ~]# echo ${name} changcheng [root@localhost ~]# echo ${name#*ang} cheng [root@localhost ~]# 格式二:${var##*word} //其中word是指定的分隔符,整个表达式的功能是:自左向右,查找var变量所存储的字符串中,最后一次出现的word分隔符,删除字符串开头至此分隔符之间的所有字符。 示例: [root@localhost ~]# mypath="/etc/init.d/functions" [root@localhost ~]# echo ${mypath} /etc/init.d/functions [root@localhost ~]# echo ${mypath#*/} etc/init.d/functions [root@localhost ~]# echo ${mypath##*/} //取路径基名 functions [root@localhost ~]# 格式三:${var%word*} //其中word是指定的分隔符,功能是:自右向左,查找var变量所存储的字符串中,第一次出现的word分隔符,删除此分隔符至字符串结尾部分之间的所有字符。 示例: [root@localhost ~]# mypath="/etc/init.d/functions" [root@localhost ~]# echo ${mypath} /etc/init.d/functions [root@localhost ~]# echo ${mypath%/*} //取路径名 /etc/init.d [root@localhost ~]# 格式四:${var%%word*} //其中word是指定的分隔符,功能是:自右向左,查找var变量所存储的字符串中,最后一次出现的word分隔符,删除此分隔符至字符串结尾部分之间的所有字符。 示例: [root@localhost ~]# mypath="/etc/init.d/functions" [root@localhost ~]# echo ${mypath} /etc/init.d/functions [root@localhost ~]# echo ${mypath%%/*} [root@localhost ~]# [root@localhost ~]# myurl=http://www.magedu.com:80 [root@localhost ~]# echo ${myurl##*:} //取端口号 80 [root@localhost ~]# echo ${myurl%%:*} //取协议号 http [root@localhost ~]# |
8.3:、字符串查找和替换: 格式一:${var/PATTERN/SUBSTI} //查找var表示的字符串中,第一次被PATTERN所匹配的字符串,将其替换为SUBSTI所表示的字符串。 格式二:${var//PARTTERN/SUBSTI} //查找var表示的字符串中,所有被PATTERN所匹配的字符串,全部将其替换为SUBSTI所表示的字符串。 格式三:${var/#PATTERN/SUBSTI} //查找var表示的字符串中,行首被PATTERN所匹配的字符串,将其替换为SUBSTI所表示的字符串;注意:仅行首被匹配时才做替换。 格式四:${var/%PATTERN/SUBSTI} //查找var表示的字符串中,行尾被PATTERN所匹配的字符串,将其替换为SUBSTI所表示的字符串;注意:仅行尾被匹配时才做替换。 注意:此处的PATTERN仅适用glob风格的通配模式。 示例: [root@localhost ~]# userinfo=`head -1 /etc/passwd` [root@localhost ~]# echo ${userinfo} root:x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo/root/user} user:x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo//root/user} user:x:0:0:user:/user:/bin/bash [root@localhost ~]# echo ${userinfo/#oot/mage} root:x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo/#roo/mage} maget:x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo/%roo/mage} root:x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo/%bas/mage} root:x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo/%bash/mage} root:x:0:0:root:/root:/bin/mage [root@localhost ~]# |
8.4、字符串的查找删除: 格式一:${var/PATTERN} //以PATTERN为模式,查找var字符串中第一次的匹配,并删除。 格式二:${var//PATTERN} //以PATTERN为模式,查找var字符串中所有匹配的字符串,并删除。 格式三:${var/#PATTERN} //以PATTERN为模式,查找var字符串中仅行首匹配到的字符串,并删除。 格式四:${var/%PATTERN} //以PATTERN为模式,查找var字符串中仅行尾匹配的字符串,并删除。 示例: [root@localhost ~]# userinfo=$(head -1 /etc/passwd) [root@localhost ~]# echo ${userinfo} root:x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo/root} :x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo//root} :x:0:0::/:/bin/bash [root@localhost ~]# echo ${userinfo/#root} :x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo/%root} root:x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo/%bash} root:x:0:0:root:/root:/bin/ [root@localhost ~]# |
8.5、字符串大小写转换: 格式一:${var^^} //把var中所有小写字符转成大写字符 格式二:${var,,} //把var中所有大写字符转成小写字符 示例: [root@localhost ~]# userinfo=$(head -1 /etc/passwd) [root@localhost ~]# echo ${userinfo} root:x:0:0:root:/root:/bin/bash [root@localhost ~]# echo ${userinfo^^} ROOT:X:0:0:ROOT:/ROOT:/BIN/BASH [root@localhost ~]# [root@localhost ~]# say="ILoveU" [root@localhost ~]# echo ${say} ILoveU [root@localhost ~]# echo ${say,,} iloveu [root@localhost ~]# |
8.6、变量赋值: 格式一:${var:-VALUE} //var是变量;表示如果var变量为空,或者未设置,那么返回VALUE,否则返回var变量的值。 格式二:${var:=VALUE} //var是变量;表示如果var变量为空,或者未设置,那么返回VALUE,并将VALUE赋值给var变量,否则返回var变量的值。 格式三:${var:+VALUE} //如果var变量不为空,则返回VALUE值。 格式四:${var:?ERROR_INFO} //如果var为空,或者未设置,那么返回ERROR_INFO为错误提示,否则,返回var值。 示例: [root@localhost ~]# echo ${xuexiao} [root@localhost ~]# echo ${xuexiao:-nanda} nanda [root@localhost ~]# echo ${xuexiao:=nanda} nanda [root@localhost ~]# echo ${xuexiao} nanda [root@localhost ~]# echo ${xuexiao:+beida} beida [root@localhost ~]# echo ${xuexiao:?buzhidao} nanda [root@localhost ~]# echo ${xuexiao1:?buzhidao} bash: xuexiao1: buzhidao [root@localhost ~]# |
9、练习:写一个脚本,实现如下功能:
①、提示用户输入一个可执行命令的名称:
②、获取此命令所依赖的所有的库文件列表(ldd),把此目录当根
③、复制命令至某目标文件(例如:/mnt/sysroot)下的对应路径中,
例如:bash,/bin/bash ==>/mnt/sysroot/bin/bash
④、复制此命令依赖到的所有库文件至目标目录下的对应路径下:
例如:/lib64/ld-linux-x86_64.so.2 ==>/mnt/sysroot//lib64/ld-linux-x86_64.so.2
⑤、每次复制完成一个命令后,不要退出,而是提示用户继续输入要复制的其他命令,并重复完成如上述所描述的功能,直到用户输入quit退出脚本。