我很清楚source
(又名.
)实用程序,它将从文件中获取内容并在当前shell中执行它们.
现在,我正在将一些文本转换为shell命令,然后运行它们,如下所示:
$ ls | sed ... | sh
ls
只是一个随机的例子,原始文本可以是任何东西.sed
也是一个转换文本的例子.有趣的是sh
.我管道我得到的任何东西sh
,它运行它.
我的问题是,这意味着启动一个新的子shell.我宁愿让命令在我当前的shell中运行.就像我能够做的那样source some-file
,如果我在文本文件中有命令.
我不想创建临时文件因为感觉很脏.
或者,我想以与当前shell完全相同的特性启动我的子shell.
好吧,使用反引号的解决方案当然有效,但我经常需要在检查和更改输出时这样做,所以我更喜欢是否有办法将结果输入到最后.
啊,/dev/stdin
事情看起来很漂亮,但是,在一个更复杂的情况下,它没有用.
所以,我有这个:
find . -type f -iname '*.doc' | ack -v '\.doc$' | perl -pe 's/^((.*)\.doc)$/git mv -f $1 $2.doc/i' | source /dev/stdin
这确保所有.doc
文件的扩展名都是小写的.
顺便说一下,可以处理xargs
,但除此之外.
find . -type f -iname '*.doc' | ack -v '\.doc$' | perl -pe 's/^((.*)\.doc)$/$1 $2.doc/i' | xargs -L1 git mv
所以,当我运行前者时,它会马上退出,没有任何反应.
该eval
命令存在于此目的.
eval "$( ls | sed... )"
更多来自bash手册:
EVAL
eval [arguments]参数连接在一起形成一个命令,然后读取并执行该命令,并将其退出状态作为eval的退出状态返回.如果没有参数或只有空参数,则返回状态为零.
$ ls | sed ... | source /dev/stdin
更新:这适用于bash 4.0,以及tcsh和dash(如果你source
改为.
).显然这是bash 3.2中的错误.从bash 4.0发行说明:
修复了导致"."的错误.无法从非常规文件(如设备或命名管道)读取和执行命令.
哇,我知道这是一个老问题,但我最近发现自己有同样的问题(这就是我到这里的方式).
无论如何 - 我不喜欢这个source /dev/stdin
答案,但我认为我发现了一个更好的答案.实际上看起来很简单:
echo ls -la | xargs xargs
很好,对吧?实际上,这仍然没有做你想要的,因为如果你有多行,它会将它们连接成一个命令,而不是分别运行每个命令.所以我找到的解决方案是:
ls | ... | xargs -L 1 xargs
该-L 1
选项意味着每个命令执行使用(最多)1行.注意:如果您的行以尾随空格结尾,它将与下一行连接!因此,请确保每一行以非空格结束.
最后,你可以做到
ls | ... | xargs -L 1 xargs -t
查看执行的命令(-t是详细的).
希望有人读到这个!
尝试使用进程替换,它将命令的输出替换为可以获取的临时文件:
source <(echo id)
我相信这是问题的"正确答案":
ls | sed ... | while read line; do $line; done
也就是说,人们可以进入while
循环; 该read
命令命令从它的一条线stdin
,并将其分配给变量$line
.$line
然后成为循环中执行的命令; 并继续直到输入中没有其他行.
这仍然不适用于某些控制结构(如另一个循环),但在这种情况下它符合要求.
`ls | sed ...`
我有点感觉象ls | sed ... | source -
是漂亮,但不幸的是source
不理解-
的意思stdin
.