shell的 关联数组 和 普通数组

前言

  • 普通数组中判断元素是否存在的时间复杂度一般为O(n),是因为大家一般的做法都是遍历一遍数组看看所要找的数据是否存在。

  • 这里介绍一种利用key的唯一性来实现的时间复杂度为O(1)的判断元素是否存在的方法:关联数组。

  • 在c语言里这种方法经常使用,这里介绍在shell的语法中我们如何利用关联数组和key值唯一性来提高脚本的执行效率


1. 普通数组

  bash支持一维数组(不支持多维数组),并且没有限定数组的大小。类似与C语言,数组元素的下标由0开始编号。获取数组中的元素要利用下标,下标可以是整数或算术表达式,其值应大于或等于0。

  1.1 定义数组

  在Shell中,用括号来表示数组,数组元素用“空格”符号分割开。定义数组的一般形式为:
    array_name=(value1 ... valuen)

  

复制代码
    array_name=(value0 value1 value2 value3)

    array_name=(
    value0
    value1
    value2
    value3
    )
复制代码

  还可以单独定义数组的各个分量:

    array_name[0]=value0
    array_name[1]=value1
    array_name[2]=value2

  可以不使用连续的下标,而且下标的范围没有限制。

  1.2 读取数组

读取数组元素值的一般格式是:
    ${array_name[index]}
例如:

  1. valuen=${array_name[2]}

举个例子:

  

复制代码
    #!/bin/sh
    NAME[0]="Zara"
    NAME[1]="Qadir"
    NAME[2]="Mahnaz"
    NAME[3]="Ayan"
    NAME[4]="Daisy"
    echo "First Method: ${NAME[*]}"
    echo "Second Method: ${NAME[@]}"
复制代码
 

运行脚本,输出:

$./test.sh
First Index: Zara
Second Index: Qadir

使用@ 或 * 可以获取数组中的所有元素,例如:

  

    ${array_name[*]}
    ${array_name[@]}

 

  举个例子:

  

复制代码
    #!/bin/sh
    NAME[0]="Zara"
    NAME[1]="Qadir"
    NAME[2]="Mahnaz"
    NAME[3]="Ayan"
    NAME[4]="Daisy"
    echo "First Method: ${NAME[*]}"
    echo "Second Method: ${NAME[@]}"
复制代码

  运行脚本,输出:

$./test.sh
First Method: Zara Qadir Mahnaz Ayan Daisy
Second Method: Zara Qadir Mahnaz Ayan Daisy

  1.3获取数组的长度

获取数组长度的方法与获取字符串长度的方法相同,例如:
    # 取得数组元素的个数
    length=${#array_name[@]}
    # 或者
    length=${#array_name[*]}
    # 取得数组单个元素的长度
    lengthn=${#array_name[n]}

2. 关联数组

   bashi没有原生的对于类似hash table的支持,不像perl或python.
下标数组元素是通过数组下标(数组下标可以是算术表达式,其结果必须是一个整数)来访问的,但是这种访问方式在表达某些关联性很强的数据时会存在限制。
shell 提供了另外一种数组,其可以使用任意的字符串作为下标(不必是整数)来访问数组元素。这种数组叫做关联数组(associative array)。
关联数组的下标和值称为键值对,它们是一一对应的关系。
在关联数组中,键是唯一的,值可以不唯一。
关联数组有时候称为哈希(hash)。

定义关联数组
shell 的关联数组和 shell 的下标数组在定义和使用上完全一样,只是在索引上有区别。
需要注意的是,在使用关联数组之前,需要使用命令 declare -A array 进行显示声明。

示例 1
test.sh 文件的内容如下
name=(jim tom lucy)              # 键数组

declare -A phone                 #定义关联数组
phone=([jim]=135 [tom]=136 [lucy]=158)

for i in `eval echo {0..$((${#name[*]}-1))}`
do
       echo ${name[i]} phone number is ${phone["${name[i]}"]}
done

在命令提示符下输入 ./test.sh,执行结果如下:
jim phone number is 135
tom phone number is 136
lucy phone number is 158

操作关联数组的语法
关联数组的操作语法和数组的操作语法完全一致,如下列出常见的操作。

 语法 描述
 ${!array[*]} 取关联数组所有键
 ${!array[@]} 取关联数组所有键
 ${array[*]} 取关联数组所有值
 ${array[@]} 取关联数组所有值
 ${#array[*]} 关联数组的长度
 ${#array[@]} 关联数组的长度

示例 2
test.sh 文件的内容如下
declare -A phone               #定义关联数组
phone=([jim]=135 [tom]=136 [lucy]=158)

for key in ${!phone[*]}
do
       echo "$key -> ${phone[$key]}"
done

在命令提示符下输入 ./test.sh,执行结果如下:
tom -> 136
jim -> 135
lucy -> 158
本,输出:

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值