正确引用bash别名定义

时间:2018-11-20 17:53:59

标签: bash alias

我有以下命令要尝试放入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

我认为这意味着我没有正确使用引号,但是在确定正确的组合时遇到了麻烦。帮助吗?

2 个答案:

答案 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"