我声明了一个变量VAR="XYZ YZA ZAB"
,我想在该变量中的所有单词旁边附加一些字符串。
例如:我想附加.file1
到 中的每个空格分隔的子字符串XYZ YZA ZAB
。
变量的输出应如下所示,不带任何for loop
内容。
VAR="XYZ.file1 YZA.file1 ZAB.file1"
我确信awk
可以使用,但我不知道如何将其用于此目的。
答案1
和ksh93
:
VAR2=${VAR//+([^[:space:]])/\1.file1}
zsh
与:相同
set -o extendedglob
VAR2=${VAR//(#m)[^[:space:]]##/$MATCH.file1}
POSIXly:
VAR2=$(printf '%s\n' "$VAR" | sed 's/[^[:space:]]\{1,\}/&.file1/g')
(请注意,如果 中存在尾随换行符,它会删除$VAR
)。
它们都将除空白字符 ( ) 之外的一个或多个 ( +(...)
, ##
, ) 字符序列替换为相同的字符 ( , , ) 并附加。\{1,\}
[^[:space:]]
\1
$MATCH
&
.file1
或者,如果您不关心保留单词之间的空白量并且单词仅由 SPC、TAB 和 NL(而不是其他空白字符)分隔,则可以拆分并连接:
unset IFS # default IFS of SPC+TAB+NL
set -o noglob # disable glob
set -- $VAR # split+glob without glob
for i do
set -- "$@" "$i.file1" # add suffix
shift
done
VAR2="$*" # join with space
对于支持数组的 shell,您可能需要使用数组变量而不是标量变量。和rc
/ es
/ zsh
/ ksh93
/ bash
/ mksh
/ yash
:
VAR=(XYZ YZA ZAB)
然后添加.file1
到每个元素(这次可能包含空格本身)只是一个问题:
VAR2=($VAR^.file1) # rc, es
VAR2=($^VAR.file1) # zsh
VAR2=("${VAR[@]/*/\0.file1}") # ksh93
VAR2=("${VAR[@]/%/.file1}") # bash
答案2
简单的解决方案是使用 shell 分割功能(默认为IFS
):
$ set -f; printf '%s.file1 ' $var; echo
XYZ.file1 YZA.file1 ZAB.file1
了解这会将重复的空格/制表符折叠为一个,并且会删除前导和尾随空白。这set -f
将避免包含全局字符 ( * ? or [
) 的文件名出现问题。在这种情况下,打印的字符串将有一个额外的空格。
答案3
对于zsh
,这是一种比 Stéphane 的方法稍微短一点的方法:
$ VAR="XYZ YZA ZAB"
$ VAR=( ${^=VAR}.file )
$ print -l $VAR
XYZ.file
YZA.file
ZAB.file
=
执行字符串的分词,然后将^
后缀“分配”到每个单词。
http://zsh.sourceforge.net/Doc/Release/Expansion.html#Parameter-Expansion