我试图找出是否可以在单个sed命令中编辑一个文件,而无需手动将编辑的内容流到一个新文件中,然后将新文件重命名为原始文件名。
我尝试了-i选项,但是Solaris系统说-i是非法选项。还有别的办法吗?
我试图找出是否可以在单个sed命令中编辑一个文件,而无需手动将编辑的内容流到一个新文件中,然后将新文件重命名为原始文件名。
我尝试了-i选项,但是Solaris系统说-i是非法选项。还有别的办法吗?
当前回答
mv file.txt file.tmp && sed 's/foo/bar/g' < file.tmp > file.txt
应该保留所有硬链接,因为输出被直接返回以覆盖原始文件的内容,并且避免了对sed的任何特殊版本的需要。
其他回答
需要注意的一点是,sed不能自己写文件,因为sed的唯一目的是充当“流”(即stdin、stdout、stderr和其他>&n缓冲区、套接字等的管道)上的编辑器。考虑到这一点,您可以使用另一个命令tee将输出写回文件。另一种选择是创建一个补丁,从管道的内容到diff。
三通法
sed '/regex/' <file> | tee <file>
修补方法
sed '/regex/' <file> | diff -p <file> /dev/stdin | patch
更新:
另外,请注意patch将从diff输出的第一行获取要更改的文件:
Patch不需要知道要访问哪个文件,因为这在diff输出的第一行中找到:
$ echo foobar | tee fubar
$ sed 's/oo/u/' fubar | diff -p fubar /dev/stdin
*** fubar 2014-03-15 18:06:09.000000000 -0500
--- /dev/stdin 2014-03-15 18:06:41.000000000 -0500
***************
*** 1 ****
! foobar
--- 1 ----
! fubar
$ sed 's/oo/u/' fubar | diff -p fubar /dev/stdin | patch
patching file fubar
在一个系统中,sed没有编辑文件的能力,我认为更好的解决方案是使用perl:
perl -pi -e 's/foo/bar/g' file.txt
虽然这确实创建了一个临时文件,但它替换了原始文件,因为已经提供了一个空的就地后缀/扩展名。
您没有指定使用什么shell,但是对于zsh,您可以使用=()构造来实现这一点。大致如下:
cp =(sed ... file; sync) file
=()类似于>(),但创建一个临时文件,当cp终止时自动删除。
非常好的例子。我在编辑许多文件时遇到了挑战,而-i选项似乎是在find命令中使用它的唯一合理解决方案。下面的脚本在每个文件的第一行前面添加“version:”:
find . -name pkg.json -print -exec sed -i '.bak' '1 s/^/version /' {} \;
就像《007:大破天幕杀机》中Moneypenny说的:“有时候旧的方法是最好的。” 金凯德后来也说过类似的话。
$ printf ',s/false/true/g\nw\n' | ed {YourFileHere}
快乐编辑到位。 增加了'\nw\n'来写入文件。很抱歉没有及时回复请求。