我有以下命令要尝试放入bash别名中。该命令本身可以正常工作,但是当我尝试使用别名时,出现以下错误:
命令
find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' \; | sort -nr
别名
alias csfiles='find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo "$(find "{}" -type f | wc -l)" {}' \; | sort -nr'
错误:
-sh: alias 0: not found
-sh: alias {} \; | sort nr: not found
我认为这意味着我没有正确使用引号,但是在确定正确的组合时遇到了麻烦。帮助吗?
答案 0 :(得分:4)
您的外部find
不能完成简单的glob无法做到的任何事情。这样就消除了一层引号(以及找到的每个目录的sh
进程)。
# Ignoring the issue of assuming no file name contains a newline
for d in ./*/; do
echo "$(find "$d" -type f | wc -l) $d"
done
只需定义一个shell函数即可消除施加在alias
的参数上的第二层。
csfiles () {
for d in ./*/; do
echo "$(find "$d" -type f | wc -l) $d"
done
}
还可以用find
循环替换对for
的其余调用,从而消除了每个文件名只占一行的问题:
csfiles () {
for d in ./*/; do
echo "$(for f in "$d"/*; do [ -f "$f" ] && echo; done | wc -l) $d"
done
}
如果find
支持-printf
主目录,则可以保留,因为您不关心文件的实际名称,而只是每个文件只得到一行输出。
csfiles () {
for d in ./*/; do
echo "$(find "$d" -type f -printf . | wc -l) $d"
done
}
答案 1 :(得分:1)
您可以在定义周围使用双引号,例如:
alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec sh -c 'echo \"\$(find \"{}\" -type f | wc -l)\" {}' \; | sort -nr"
定义中的每个文字"
都会被转义:\"
。
注意:您还需要转义内部命令替换,以防止其在别名定义时扩展。像这样... \$(...)
作为对作弊者评论的后续,您应该将文件名作为参数传递给内部find命令。否则,如果其中一个文件夹的名称中带有"
,则会遇到问题:
alias foo="find . -maxdepth 1 -mindepth 1 -type d -exec bash -c 'echo \"\$(find \"\${1}\" -type f | wc -l) \"\${1}\" \"' -- \"{}\" \; | sort -nr"