(4)shell脚本编程进阶之数组、字符串切片

(一)数组定义

  • 数组:存储多个元素的连续的内存空间,相当于多个变量的集合
  • 数组名和索引
  • 索引:编号从0 开始,属于数值索引
  • 注意:索引可支持使用自定义的格式,而不仅是数值格式,即为关联索引, ,bash4.0版本之后开始支持
  • bash 的数组支持稀疏格式(索引不连续)
  • 声明数组:
    declare -a ARRAY_NAME
    declare -A ARRAY_NAME: 关联数组
    注意:两者不可相互转换
  • (1)数组赋值
    • 一次只赋值一个元素;
      ARRAY_NAME[INDEX]=VALUE
      weekdays[0]="Sunday"
      weekdays[4]="Thursday"
    • (一次赋值全部元素:
      ARRAY_NAME=("VAL1" "VAL2" "VAL3" ...)
    • 只赋值特定元素:
      ARRAY_NAME=([0]="VAL1" [3]="VAL2" ...)
    • 交互式数组值对赋值
      read -a ARRAY

(二)引用数组

  • 引用数组元素:${ARRAY_NAME[INDEX]}注意:省略[INDEX] 表示引用下标为0 的元素
  • 数组的长度( 数组中元素的个数):
    ${#ARRAY_NAME[*]}
    ${#ARRAY_NAME[@]}

示例:生成10 个随机数保存于数组中,并找出其最大值和最小值
#!/bin/bash
declare -a rand
declare -i max=0
declare –i min=32767
for i in {0..9}; do
rand[$i]=$RANDOM
echo ${rand[$i]}
[ ${rand[$i]} -gt $max ]&& max=${rand[$i]}
[ ${rand[$i]} -lt $min ]&& min=${rand[$i]}
done
echo "Max: $max Min:$min"

编写脚本,定义一个数组,数组中的元素是/var/log 目录下所有以
.log 结尾的文件;要统计其下标为偶数的文件中的行数之和
#!/bin/bash
declare -a files
files=(/var/log/.log)
declare -i 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."

  • (3)数组数据处理
    • 引用数组中的元素:
      所有元素:${ARRAY[@]}, ${ARRAY[*]}
    • 数组切片:${ARRAY[@]:offset:number}
      • offset: 要跳过的元素个数
      • number: 要取出的元素个数
      • 取偏移量之后的所有元素
        ${ARRAY[@]:offset}
    • 向数组中追加元素:
      ARRAY[${#ARRAY[*]}]
    • 删除数组中的某元素:导致稀疏格式
      unset ARRAY[INDEX]
    • 删除整个数组:unset ARRAY
    • 关联数组:
      declare -A ARRAY_NAME 注意:必须先声明,再调用
      ARRAY_NAME=([idx_name1]='val1' [idx_name2]='val2‘...)

(三)字符串切片

(1)字符串处理切片
  • ${#var}: 返回字符串变量var 的长度
  • ${var:offset}: 返回 字符串变量var 中从第offset个字符后(不包括第offset个字符)的字符开始,到最后的部分,offset的取值在0 到${#var}-1之间(bash4.2 后,允许为负值)
  • ${var:offset:number} :返回 字符串变量var 中从第offset个字符后(不包括第offset个字符)的字符开始 ,长度为number 的部分
  • ${var: -length}:取字符串的最右侧几个字符注意:冒号后必须有一空白字符
  • ${var:offset:-length}:从最左侧跳过offset 字符,一直向右取到距离最右侧lengh 个字符之前的内容
  • ${var: -length:-offset}:先从最右侧向左取到length个字符开始,再向右取到距离最右侧offset个字符之间的内容
(2)字符串处理
  • (1)基于模式取子串

  • ${var#*word}:其中word 可以是指定的任意字符

  • 功能:自左而右,查找var 变量所存储的字符串中,第一次出现的word, 删除字符串开头至第一次出现word 字符之间的所有字符

  • ${var##*word}:同上,贪婪模式,不同的是,删除的是字符串开头至最后一次由word 指定的字符之间的所有内容

     示例:
     file="var/log/messages“
     ${file#*/}: log/messages
     ${file##*/}: messages
    
  • ${var%word*}:其中word 可以是指定的任意字符;

  • 功能:自右而左,查找var 变量所存储的字符串中,第一次出现的word, 删除字符串最后一个字符向左至第一次出现word 字符之间的所有字符;
    file="/var/log/messages"
    ${file%/*}: /var/log

  • ${var%%word*}:同上,只不过删除字符串最右侧的字符向左至最后一次出现word 字符之间的所有字符;
    示例:
    url=http://www.magedu.com:80
    ${url##:} 80
    ${url%%:
    } http

(2)字符串查找替换
  • ${var/pattern/substr}:查找var 所表示的字符串中,第一次被pattern 所匹配到的字符串,以substr替换之

  • ${var//pattern/substr}: 查找var所表示的字符串中,所有能被pattern所匹配到的字符串,以substr替换之

  • ${var/#pattern/substr}:查找var 所表示的字符串中,行首被pattern 所匹配到的字符串,以substr替换之

  • ${var/%pattern/substr}:查找var 所表示的字符串中,行尾被pattern所匹配到的字符串,以substr 替换之

  • ${var/pattern}:查找var所表示的字符串中,删除第一次被pattern所匹配到的字符串

    ${var//pattern} :所有
    ${var/#pattern} :行首
    ${var/%pattern} :行尾
    
  • 字符大小写转换

  • ${var^^}:把var 中的所有小写字母转换为大写

  • ${var,,}:把var中的所有大写字母转换为小写

  • 变量赋值

  • ${var:-value}:如果var 为空或未设置,那么返回value;否则返回var 的值,可省略:

  • ${var:+value}:如果var不空,则返回value ,否则返回空值

  • ${var:=value} :如果var 为空或未设置,那么返回value,并将value 赋值给var;否则返回var的值

  • ${var:?error_info}:如果var为空或未设置,那么在当前终端打error_info ;否则返回var 的值

  • 为脚本程序使用配置文件, 实现变量赋值

  • (1) 定义文本文件,每行定义“name=value”

  • (2) 在脚本中source此文件即可

(三)高级变量用法-有类型变量

(1)declare命令

Shell 变量一般是无类型的,但是bash Shell 提供了declaretypeset
declare [ 选项] 变量名
-r 声明或显示只读变量
-i 将变量定义为整型数
-a 将变量定义为数组
-A 将变量定义为关联数组
-f 显示此脚本前定义过的所有函数名及其内容
-F 仅显示此脚本前定义过的所有函数名
-x 声明或显示环境变量和函数
-l 声明变量为小写字母
declare –l var=UPPER
-u 母 声明变量为大写字母
declare –u var=lower

(2)eval 命令
  • 定义:eval 命令将会首先扫描命令行进行所有的置换,然后再执行该命令。该命令适用于那些一次扫描无法实现其功能的变量.该命令对变量进行两次扫描
    [root@server ~]# CMD=whoami
    [root@server ~]# echo $CMD
    whoami
    [root@server ~]# eval $CMD
    root
    [root@server ~]# n=10
    [root@server ~]# echo {0..$n}
    {0..10}
    [root@server ~]# eval echo {0..$n}
    0 1 2 3 4 5 6 7 8 9 10
(3)间接变量引用
  • 如果第一个变量的值是第二个变量的名字,从第一个变量引用第二个变量的值就称为间接变量引用
  • variable1 的值是variable2,而variable2 又是变量名,variable2 的值为value ,间接变量引用是指通过variable1获得变量值value 的行为
    variable1=variable2
    variable2=value
  • bash Shell 提供了两种格式实现间接变量引用
    eval tempvar=$$variable1
    tempvar=${!variable1}
    示例:
    [root@server ~]# N=NAME
    [root@server ~]# NAME=wangnan
    [root@server ~]# N1=${!N}
    [root@server ~]# echo $N1
    wangnan
    [root@server ~]# eval N2=$$N
    [root@server ~]# echo $N2
    wangxnan
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值