split string to array in bash
和AS中的String.split不同,Bash没有类似的语法,但它实在是太灵活了,有很多种方式可以做类似的事情。
以前我写过一篇 Bash数组操作教程 ,今天使用Bash做文件处理的时候,发现有必要再写一篇将字符串拆分成数组的教程。但发现有人已经写了更好的教程在前面了,于是就偷懒转过来好了。
下面的内容转自Bash @ Linux
以空白作为分隔符来拆分字符串构造数组
ARR=($STR)
注意$STR不能加引号。
1 2 3 4 | STR="Hello World" ARR=($STR) declare -p ARR declare -a ARR='([0]="Hello" [1]="World")' |
用指定分隔符来拆分字符串构造数组
如果分隔符不是空白,而是别的,那么需要借助IFS变量。
default IFS (Internal Field Separator, which is space/tab/new line)
1 2 3 4 5 | echo $PATH /usr/kerberos/sbin:/usr/kerberos/bin:/usr/apache/apache-ant-1.7.1/bin:/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/root/bin IFS=: DIRS=($PATH) declare -p DIRS declare -a DIRS='([0]="/usr/kerberos/sbin" [1]="/usr/kerberos/bin" [2]="/usr/apache/apache-ant-1.7.1/bin" [3]="/usr/local/sbin" [4]="/usr/local/bin" [5]="/sbin" [6]="/bin" [7]="/usr/sbin" [8]="/usr/bin" [9]="/root/bin")' |
但是下面的方式是不行的。
1 2 3 4 5 | echo $PATH /usr/kerberos/sbin:/usr/kerberos/bin:/usr/apache/apache-ant-1.7.1/bin:/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/root/bin IFS=: declare -a DIRS=($PATH) declare -p DIRS declare -a DIRS='([0]="/usr/kerberos/sbin:/usr/kerberos/bin:/usr/apache/apache-ant-1.7.1/bin:/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/root/bin")' |
使用read -a来拆分字符串构造数组
All work and no play makes Jack a dull boy.
只会用功不玩耍,聪明孩子也变傻。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | PROVERB&#61;"All work and no play makes Jack a dull boy." read -a WORDS <<<$PROVERB echo "$WORDS" All echo "${#WORDS}" 3 echo "${WORDS[*]}" All work and no play makes Jack a dull boy. echo "${WORDS[&#64;]}" All work and no play makes Jack a dull boy. echo "${#WORDS[*]}" 10 echo "${#WORDS[&#64;]}" 10 |
前面的例子中要分割的字符串是以空格分割的&#xff0c;现在举一个以:分割的例子。
1 2 3 4 5 | echo $IFS IFS&#61;: read -r -a DIRS <<<"$PATH" echo $IFS declare -p DIRS declare -a DIRS&#61;&#39;([0]&#61;"/usr/kerberos/sbin" [1]&#61;"/usr/kerberos/bin" [2]&#61;"/usr/apache/apache-ant-1.7.1/bin" [3]&#61;"/usr/local/sbin" [4]&#61;"/usr/local/bin" [5]&#61;"/sbin" [6]&#61;"/bin" [7]&#61;"/usr/sbin" [8]&#61;"/usr/bin" [9]&#61;"/root/bin")&#39; |
下面的例子是将当前工作目录以/进行分割。
1 2 3 4 5 6 7 8 | echo $PWD /root/work191/ct08/src/ctmw IFS&#61;/ read -r -a PARTS <<<$PWD declare -p PARTS declare -a PARTS&#61;&#39;([0]&#61;" root work191 ct08 src ctmw")&#39; IFS&#61;/ read -r -a PARTS <<<"$PWD" declare -p PARTS declare -a PARTS&#61;&#39;([0]&#61;"" [1]&#61;"root" [2]&#61;"work191" [3]&#61;"ct08" [4]&#61;"src" [5]&#61;"ctmw")&#39; |
使用cut命令分隔字符串
1 | echo "$STR" | cut -f $N |
以TAB分隔&#xff0c;打印第N个子串值&#xff0c;N从1开始计数。
1 | echo "$STR" | cut -d "$DELIM" -d $N |
以指定DELIM分隔&#xff0c;打印第N个子串值&#xff0c;N从1开始计数。
其中&#xff0c;-d部分也可以是$N1,$N2,$N3的形式&#xff0c;即输出多个子串。
A good beginning is half done.
良好的开端是成功的一半。
1 2 3 4 5 6 7 8 9 | STR&#61;"A good beginning is half done." echo $STR | cut -d &#39; &#39; -f 2 good echo $PATH /usr/kerberos/sbin:/usr/kerberos/bin:/usr/apache/apache-ant-1.7.1/bin:/usr/local/sbin:/usr/local/bin:/sbin:/bin:/usr/sbin:/usr/bin:/root/bin echo $PATH | cut -d &#39;:&#39; -f 3 /usr/apache/apache-ant-1.7.1/bin echo $PATH | cut -d &#39;:&#39; -f 3,5 /usr/apache/apache-ant-1.7.1/bin:/usr/local/bin |
使用awk命令分隔字符串
1 2 | echo "$STR" | awk &#39;{print $1}&#39; echo "$STR" | awk &#39;{print $2}&#39; |
注意&#xff1a;awk后面的参数用单引号&#xff0c;不能用双引号。
1 2 3 4 5 6 7 | STR&#61;"A good beginning is half done." echo "$STR" | awk &#39;{print $1}&#39; A echo "$STR" | awk &#39;{print $2}&#39; good echo "$STR" | awk &#39;{print $5,$6}&#39; half done. |