13

假设有一个文本文件,a.txt例如

啊啊啊
bbb
ccc
ddd

我需要myprefix_在文件中的每一行添加一个前缀(例如):

myprefix_aaa
myprefix_bbb
myprefix_ccc
myprefix_ddd

我可以这样做awk

awk '{print "myprefix_" $0}' a.txt

现在我想知道是否有另一种方法可以在 shell 中做到这一点。

4

4 回答 4

23

sed

$ sed 's/^/myprefix_/' a.txt
myprefix_aaa
myprefix_bbb
myprefix_ccc
myprefix_ddd

这将替换以 开头^的每一行myprefix_。请注意,^不会丢失,因此这允许将内容添加到每行的开头。

你可以使你awk的版本更短:

$ awk '$0="myprefix_"$0' a.txt
myprefix_aaa
myprefix_bbb
myprefix_ccc
myprefix_ddd

或传递值:

$ prefix="myprefix_"
$ awk -v prefix="$prefix" '$0=prefix$0' a.txt
myprefix_aaa
myprefix_bbb
myprefix_ccc
myprefix_ddd

也可以通过以下方式完成nl

$ nl -s "prefix_" a.txt | cut -c7-
prefix_aaa
prefix_bbb
prefix_ccc
prefix_ddd

最后:正如John Zwinck 解释的那样,你也可以这样做:

paste -d'' <(yes prefix_) a.txt | head -n $(wc -l a.txt)

在 OS X 上:

paste -d '\0' <(yes prefix_) a.txt | head -n $(wc -l < a.txt)
于 2013-11-07T09:11:07.700 回答
15

纯重击:

while read line
do
    echo "prefix_$line"
done < a.txt
于 2013-11-07T09:24:15.447 回答
12

作为参考,关于这个问题的awksedbash解决方案的速度:

生成一个 800K 的输入文件bash

line="12345678901234567890123456789012345678901234567890123456789012345678901234567890"
rm a.txt
for i in {1..10000} ; do
    echo $line >> a.txt
done

然后考虑bash脚本timeIt

if [ -e b.txt ] ; then
    rm b.txt
fi
echo "Bash:"
time bashtest
rm b.txt
echo
echo "Awk:"
time awktest
rm b.txt
echo
echo "Sed:"
time sedtest

bashtest在哪里

while read line
do
    echo "prefix_$line" >> b.txt
done < a.txt

awktest是:

awk '$0="myprefix_"$0' a.txt > b.txt

并且sedtest是:

sed 's/^/myprefix_/' a.txt > b.txt

我在我的机器上得到了以下结果:

Bash:

real    0m0.401s
user    0m0.340s
sys 0m0.048s

Awk:

real    0m0.009s
user    0m0.000s
sys 0m0.004s

Sed:

real    0m0.009s
user    0m0.000s
sys 0m0.004s

似乎bash解决方案要慢得多..

于 2013-11-07T12:32:25.703 回答
2

您还可以使用该xargs实用程序:

cat file | xargs -d "\n" -L1 echo myprefix_ 

-d选项用于允许输入行带有尾随空格(与-L规范相关)。

于 2018-11-15T09:07:59.463 回答