19 Star 185 Fork 90

baoke / Linux学完就忘系列

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
该仓库未声明开源许可证文件(LICENSE),使用请关注具体项目描述及其代码上游依赖。
克隆/下载
41.shell数组和字符串处理工具.md 7.49 KB
一键复制 编辑 原始数据 按行查看 历史

数组

存储多个元素的连续的内存空间;
数组名:整个数组只有一个名字;
数组索引:编号从0开始;
数组名[索引]
${ARRAY_NAME[INDEX]}

注意:bash-4及之后的版本,支持自定义索引格式,而不仅仅是0,1,2,...数字格式;
此类数组称之为"关联数组"

#查看bash的版本
rpm -q bash

声明数组:
declare -a NAME:声明索引属组;
declare -A NAME:声明关联属组;

数组中元素的赋值方式:
(1) 一次只赋值一个元素;
ARRAY_NAME[index]=value

(2) 一次赋值全部元素;
ARRAY_NAME=("VAL1" "VAL2" "VAL3" ...)

(3) 只赋值特定元素(稀疏格式的属组);
ARRAY_NAME=([0]="VAL1" [3]="VAL4" ...)
注意:bash支持稀疏格式的数组;

(4) read -a ARRAY_NAME
-a:后跟一个变量,该变量会被认为是个数组,然后给其赋值,默认是以空格为分割符

引用数组中的元素:${ARRAY_NAME[INDEX]}
注意:引用时,只给数组名,表示引用下标为0的元素;

数组的长度(数组中元素的个数):
${#ARRAY_NAME[*]}

echo ${#jianghu[*]}
echo ${#jianghu[@]}

示例:生成10个随机数,并找出其中的最大值和最小值;

declare -a rand
max=0

for i in $(seq 1 10);do
    rand[$i]=$RANDOM
    [ ${rand[$i]} -gt $max ] && max=${rand[$i]}
    echo ${rand[$i]}
done

echo "Max:$max"

练习:
1.生成10个随机数,而后由小到大进行排序;

2.写一个脚本
定义一个数组,数组中的元素是/var/log目录下所有以.log结尾的文件;统计其下标为偶数的文件中的行数之和;
files=(/var/log/*.log)会解析为数组,每个文件以空格分隔;

#!/bin/bash

declare -a files

files=(/var/log/*.log)

lines=0

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}"

引用数组中的所有元素:
${ARRAY_NAME[*]}
${ARRAY_NAME[@]}

echo ${jianghu[*]}  
echo ${jianghu[@]}  

数组元素切片:${ARRAY_NAME[*]:offset:number}
offset:偏移量(指定数组下标);
number:要取出的元素个数;省略number时,表示取偏移量之后的所有元素;

files=(/etc/[Pp]*)
#从数组下标2开始取3个元素
echo ${files[*]:2:3}
#从数组下标5开始取之后所有元素
echo ${files[*]:5}

向非稀疏格式数组中追加元素:
ARRAY_NAME[${#ARRAY_NAME[*]}]=VALUE

删除数组中的某元素:
unset ARRAY[INDEX]
注意:撤销元素不是引用属组中的值,所以这里不需要加${}

关联数组:
必须先声明数组:
declare -A ARRAY_NAME
ARRAY_NAME=([index_name1]="value1" [index_name2]="value2")

bash的内置字符串处理工具

字符串切片:
${var:offset:number}
取字符串的子串;
取字符串最右侧的几个字符:${var: -length};注意:冒号后必须有一个空白字符;

name=Jerry
#截取索引从2开始后所有字符
echo ${name:2}
#截取索引从2开始后2位的字符
echo ${name:2:2}
#从右侧向前取4个字符,正向截取从索引0开始,反向从索引-1开始,最后一个字符为-1
echo ${name: -4}

基于模式取子串:
${var#*word}:其中word是指定的分隔符;功能:自左而右,查找var变量所存储的字符串中,第一次出现的word分隔符,删除字符串开头至此分隔符之间的所有字符;
${var##*word}:其中word是指定的分隔符;功能:自左而右,查找var变量所存储的字符串中,最后一次出现的word分隔符,删除字符串开头至此分隔符之间的所有字符;

mypath="/etc/init.d/functions"
echo ${mypath#*/}
echo ${mypath##*/}

${var%word*}:其中word是指定的分隔符;功能:自右而左,查找var变量所存储的字符串中,第一次出现的word分隔符,删除此分隔符至字符串尾部之间的所有字符;
${var%%word*}:其中word是指定的分隔符;功能:自右而左,查找var变量所存储的字符串中,最后一次出现的word分隔符,删除此分隔符至字符串尾部之间的所有字符;

mypath="/etc/init.d/functions"
echo ${mypath%/*}
mypath="etc/init.d/functions"
echo ${mypath%%/*}

url=http://www.baidu.co,:80
#取端口号
echo ${url##*:}
#取协议
echo ${url%%:*}

查找替换:
${var/PATTERN/SUBSTI}:查找var所表示的字符串中,第一次被PATTERN所匹配到的字符串,将其替换为SUBSTI所表示的字符串;
${var//PATTERN/SUBSTI}:查找var所表示的字符串中,所有被PATTERN所匹配到的字符串,并将其全部替换为SUBSTI所表示的字符串;

${var/#PATTERN/SUBSTI}:查找var所表示的字符串中,行首被PATTERN所匹配到的字符串,将其替换为SUBSTI所表示的字符串;
${var/%PATTERN/SUBSTI}:查找var所表示的字符串中,行尾被PATTERN所匹配到的字符串,将其替换为SUBSTI所表示的字符串;

注意:这里的PATTERN只能使用GLOB风格的通配符

userinfo="root:x:0:0:root admin:/root:/bin/chroot"
echo ${userinfo/r??t/ROOT}
echo ${userinfo//r??t/ROOT}
echo ${userinfo/#r??t/ROOT}
echo ${userinfo/%r??t/ROOT}

查找删除:
${var/PATTERN}:以PATTERN为模式查找var字符串中第一次的匹配,并删除之;
${var//PATTERN}:以PATTERN为模式查找var字符串中,所有被PATTERN所匹配的字符串,并删除之;
${var/#PATTERN}:以PATTERN为模式查找var字符串中,行首被PATTERN所匹配到的字符串,并删除之;
${var/%PATTERN}:以PATTERN为模式查找var字符串中,行尾被PATTERN所匹配到的字符串,并删除之;

字符串大小写转换:
${var^^}:把var中的所有小写字符转换为大写;
${var,,}:把var中的所有大写字符转换为小写;

url=http://www.baidu.com:80
echo ${url^^}
myurl=${url^^}
echo ${url,,}

变量赋值:
${var:-VALUE}:如果var变量为空,或未设置,那么返回VALUE;否则,则返回var变量的值;
${var:=VALUE}:如果var变量为空,或未设置,那么返回VALUE,并将VALUE赋值给var变量;否则,则返回var变量的值;
${var:+VALUE}:如果var变量不空,则返回VALUE;
${var:?ERROR_INFO}:如果var为空,或未设置,那么返回ERROR_INFO为错误提示;否则,返回var值;

练习:写一个脚本,完成如下功能
(1) 提示用户输入一个可执行命令的名称;
(2) 获取此命令所依赖到的所有库文件列表;
(3) 复制命令至某目标目录(例如/mnt/sysroot,即把此目录当作根)下的对应的路径中
bash,/bin/bash ==> /mnt/sysroot/bin/bash
useradd,/usr/sbin/useradd ==> /mnt/sysroot/usr/sbin/useradd

#查找可执行命令所在的目录
whereis ls
whereis useradd
#显示依赖库文件的目录
ldd /bin/ls
ldd /usr/sbin/useradd

(4) 复制此命令依赖到的所有库文件至目标目录下的对应路径下:
/lib64/ld-linux-x8664.so.2 ==> /mnt/sysroot/lib64/ld-linux-x8664.so.2

进一步:
每次复制完成一个命令后,不要退出,而是提示用户继续输入要复制的其它命令,并重复完成如上所描述的功能;直到用户输入"quit"退出脚本;

Shell
1
https://gitee.com/abcd1234567_1/linux.git
git@gitee.com:abcd1234567_1/linux.git
abcd1234567_1
linux
Linux学完就忘系列
master

搜索帮助