假设有一个文本文件,a.txt
例如
啊啊啊 bbb ccc ddd
我需要myprefix_
在文件中的每一行添加一个前缀(例如):
myprefix_aaa myprefix_bbb myprefix_ccc myprefix_ddd
我可以这样做awk
:
awk '{print "myprefix_" $0}' a.txt
现在我想知道是否有另一种方法可以在 shell 中做到这一点。
与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)
纯重击:
while read line
do
echo "prefix_$line"
done < a.txt
作为参考,关于这个问题的awk
、sed
和bash
解决方案的速度:
生成一个 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
解决方案要慢得多..
您还可以使用该xargs
实用程序:
cat file | xargs -d "\n" -L1 echo myprefix_
该-d
选项用于允许输入行带有尾随空格(与-L
规范相关)。