江源

2016-03-29 21:57

bash 的字符串和数组

本文作者:IMWeb 江源 原文出处:IMWeb社区 未经同意,禁止转载

原文

回顾下自己接触过的编程语言,字符串数组真是基础中的基础。也因此,在接触一门新的语言的时候,非常有必要去熟悉该语言的字符串数组

字符串

声明和赋值

name='jero' # 单引号
hellojero="hello, $name" # 双引号,可以引用变量

再看看复杂点的操作。

表达式 含义
${var} 变量 var 的值,与 $var 相同
${var-DEFAULT} 如果 var 没有被声明,那么就以 $DEFAULT 作为其值
${var:-DEFAULT} 如果 var 没有被声明, 或者其值为空, 那么就以 $DEFAULT 作为其值
${var=DEFAULT} 如果 var 没有被声明,那么就以 $DEFAULT 作为其值
${var:=DEFAULT} 如果 var 没有被声明, 或者其值为空, 那么就以 $DEFAULT 作为其值
${var+OTHER} 如果 var 声明了, 那么其值就是 $OTHER, 否则就为 null 字符串
${var:+OTHER} 如果 var 被设置了, 那么其值就是 $OTHER, 否则就为 null 字符串
${var?ERR_MSG} 如果 var 没被声明, 那么就打印 $ERR_MSG
${var:?ERR_MSG} 如果 var 没被设置, 那么就打印 $ERR_MSG
${!varprefix*} 匹配之前所有以 varprefix 开头进行声明的变量
${!varprefix@} 同上

${var:-DEFAULT} vs ${var:=DEFAULT}

这两个真的非常像,那么它们的区别是什么呢?

foo1=${bar1:-'abc'}
echo $foo1; # abc
echo $bar1; # 空

foo2=${bar2:='abc'}
echo $foo2; # abc
echo $bar2; # abc

# 区别很明显了,bar1 没有赋值, bar2 被赋值了

操作

长度

echo ${#name} #4

# 有些资料上会有如下方式:
echo `expr length "$name"`
# 在 mac(10.11.4) 上,会 `syntax error`
# 这是因为 GUN 的 expr 和 BSD 的 expr 有很大的不同,用的时候一定要慎重
# 具体可参考 https://discussions.apple.com/thread/923299

# 可做如下尝试:
echo `expr "$name" : '.*'` # 4

拼接

拼接就比较灵活了,但不外乎如下几种方式。

hei="hei, $name" # 利用双引号
family='jiang'
echo "$hei $family" # hei, jero jiang
echo $hei $family # 当然也可以不用双引号,但是可读性降低了

子字符串

表达式 含义
${#string} $string 的长度
${string:position} $string 中, 从位置 $position 开始提取子串
${string:position:length} $string 中, 从位置 $position 开始提取长度为 $length 的子串
${string#substring} 从变量 $string 的开头, 删除最短匹配 $substring 的子串
${string##substring} 从变量 $string 的开头, 删除最长匹配 $substring 的子串
${string%substring} 从变量 $string 的结尾, 删除最短匹配 $substring 的子串
${string%%substring} 从变量 $string 的结尾, 删除最长匹配 $substring 的子串
${string/substring/replacement} 使用 $replacement, 来代替第一个匹配的 $substring
${string//substring/replacement} 使用 $replacement, 代替所有匹配的 $substring
${string/#substring/replacement} 如果 $string 的前面数位匹配 $substring, 那么就用 $replacement 来代替匹配到的 $substring
${string/%substring/replacement} 如果 $string 的后面数位匹配 $substring, 那么就用 $replacement 来代替匹配到的 $substring

substring 可以是正则。

下面来个实例。

long='123456789abcdefg123456789abcdefg123456789'
echo ${#long} # 41,长度
echo ${long:3} # 456789abcdefg123456789abcdefg123456789,从第三位开始截取
echo ${long:3:6} # 456789,从第三位开始截取长度为 6 的子串
echo ${long#*a} # bcdefg123456789abcdefg123456789,从头删除最短匹配
echo ${long##*a} # bcdefg123456789,从头删除最长匹配
echo ${long%a*} # 123456789abcdefg123456789,从尾开始删除最短匹配
echo ${long%%a*} # 123456789,从尾开始删除最长匹配
echo ${long/abc/ABC} # 123456789ABCdefg123456789abcdefg123456789,替换首次出现的子字符串
echo ${long//abc/ABC} # 123456789ABCdefg123456789ABCdefg123456789,全局替换
echo ${long/#123/ABC} # ABC456789abcdefg123456789abcdefg123456789,字符串的前面数位和 123 匹配,替换之
echo ${long/%789/ABC} # 123456789abcdefg123456789abcdefg123456ABC,字符串的后面数位和 789 匹配,替换之

数组

bash 支持一维数组,下标从 0 开始。

declare -a array_name # 声明一个数组,但并没人这么用

arr1=(value0 value1 value2 value3 value)
# 或者
arr1=(
    value0
    value1
    value2
    value3
    value4
) # !!!元素之间没有逗号


# 数组元素不用连续
arr2[0]=v0,
arr2[100]=v100

# 取数组元素,必须用大括号模式
echo ${arr1[3]} # value2
echo ${arr2[100]} # v100

# 使用 `@` 或 `*` 可以获取数组中的所有元素
echo ${arr1[*]} # value0 value1 value2 value3 value4
echo ${arr2[@]} # v1 v100

# 获取数组的长度
echo ${#arr1[*]} # 5
echo ${#arr2[@]} # 2





参考文章

1条评论

    您需要 注册 一个IMWeb账号或者 才能进行评论。