我将此字符串存储在变量中:

IN="bla@some.com;john@home.com"

现在,我想将字符串拆分为:;分隔符,以便我可以:

ADDR1="bla@some.com"
ADDR2="john@home.com"

我不一定需要ADDR1和ADDR2变量。如果它们是数组的元素,那就更好了。


根据以下答案的建议,我最终得出了以下结论,这就是我想要的:

#!/usr/bin/env bash

IN="bla@some.com;john@home.com"

mails=$(echo $IN | tr ";" "\n")

for addr in $mails
do
    echo "> [$addr]"
done

输出:

> [bla@some.com]
> [john@home.com]

有一个解决方案涉及将Internal_field_separator(IFS)设置为;。我不确定这个答案发生了什么,如何将IFS重置为默认值?

回复:IFS解决方案,我尝试了这个,它奏效了,我保留了旧的IFS,然后恢复它:

IN="bla@some.com;john@home.com"

OIFS=$IFS
IFS=';'
mails2=$IN
for x in $mails2
do
    echo "> [$x]"
done

IFS=$OIFS

顺便说一下,当我尝试

mails2=($IN)

在循环中打印时,我只得到了第一个字符串,$in周围没有括号,它可以工作。


当前回答

也许不是最优雅的解决方案,但可以使用*和空格:

IN="bla@so me.com;*;john@home.com"
for i in `delims=${IN//[^;]}; seq 1 $((${#delims} + 1))`
do
   echo "> [`echo $IN | cut -d';' -f$i`]"
done

输出

> [bla@so me.com]
> [*]
> [john@home.com]

其他示例(开头和结尾的分隔符):

IN=";bla@so me.com;*;john@home.com;"
> []
> [bla@so me.com]
> [*]
> [john@home.com]
> []

基本上,它会删除除以下字符以外的所有字符:;制作熟食;;;。然后,它执行从1到${#delims}计数的分隔符数的循环。最后一步是使用切割安全地获得$i部分。

其他回答

这也适用于:

IN="bla@some.com;john@home.com"
echo ADD1=`echo $IN | cut -d \; -f 1`
echo ADD2=`echo $IN | cut -d \; -f 2`

小心,这个解决方案并不总是正确的。万一你通过了“bla@some.com“仅,它将分配给ADD1和ADD2。

您可以设置内部字段分隔符(IFS)变量,然后让它解析为数组。当在命令中发生这种情况时,IFS的赋值只发生在单个命令的环境中(读取)。然后,它根据IFS变量值将输入解析为数组,然后我们可以对其进行迭代。

此示例将解析一行由;,将其推入阵列:

IFS=';' read -ra ADDR <<< "$IN"
for i in "${ADDR[@]}"; do
  # process "$i"
done

另一个示例用于处理$IN的整个内容,每次一行输入由;:

while IFS=';' read -ra ADDR; do
  for i in "${ADDR[@]}"; do
    # process "$i"
  done
done <<< "$IN"

有一种简单而聪明的方法:

echo "add:sfff" | xargs -d: -i  echo {}

但是您必须使用gnu-xargs,BSD-xargs不能支持-d delim。如果你像我一样使用苹果mac,你可以安装gnu-xargs:

brew install findutils

then

echo "add:sfff" | gxargs -d: -i  echo {}

取自Bash shell脚本拆分数组:

IN="bla@some.com;john@home.com"
arrIN=(${IN//;/ })
echo ${arrIN[1]}                  # Output: john@home.com

说明:

此构造将替换所有出现的“;”(首字母//表示全局替换)在字符串in中使用“”(单个空格),然后将空格分隔的字符串解释为数组(这就是周围的括号所做的)。

大括号内用于替换每个“;”的语法带有“”字符的字符称为参数扩展。

有一些常见的陷阱:

如果原始字符串有空格,则需要使用IFS:

IFS=“:”;arrIN=($IN);未设置IFS;

如果原始字符串包含空格,并且分隔符是新行,则可以使用以下命令设置IFS:

IFS=$'\n';arrIN=($IN);未设置IFS;

对达伦的回答有不同的看法,我是这样做的:

IN="bla@some.com;john@home.com"
read ADDR1 ADDR2 <<<$(IFS=";"; echo $IN)