有没有办法找到执行我在shell中定义的函数?

例如:

dosomething () {
  echo "Doing something with $1"
}
find . -exec dosomething {} \;

其结果是:

find: dosomething: No such file or directory

有没有办法让find's -exec看到做某事?


当前回答

Jac的回答很好,但它有几个容易克服的陷阱:

find . -print0 | while IFS= read -r -d '' file; do dosomething "$file"; done

这将使用null作为分隔符而不是换行符,因此具有换行符的文件名将正常工作。它还使用-r标志来禁用反斜杠转义,如果没有它,文件名中的反斜杠将不起作用。它还清除IFS,以便名称中潜在的尾随空白不会被丢弃。

其他回答

我会完全避免使用-exec。使用xargs:

find . -name <script/command you're searching for> | xargs bash -c

Jac的回答很好,但它有几个容易克服的陷阱:

find . -print0 | while IFS= read -r -d '' file; do dosomething "$file"; done

这将使用null作为分隔符而不是换行符,因此具有换行符的文件名将正常工作。它还使用-r标志来禁用反斜杠转义,如果没有它,文件名中的反斜杠将不起作用。它还清除IFS,以便名称中潜在的尾随空白不会被丢弃。

不直接,不。Find在一个单独的进程中执行,而不是在您的shell中执行。

创建一个shell脚本,完成与函数相同的工作,并找到can -exec。

让脚本调用自身,将找到的每个项作为参数传递:

#!/bin/bash

if [ ! $1 == "" ] ; then
   echo "doing something with $1"
   exit 0
fi

find . -exec $0 {} \;

exit 0

当您单独运行脚本时,它会找到您正在寻找的内容,并调用自己,将每个查找结果作为参数传递。当脚本使用参数运行时,它会执行参数上的命令,然后退出。

对于那些正在寻找一个Bash函数,将在当前目录下的所有文件上执行给定命令的人,我从上面的答案中编译了一个:

toall(){
    find . -type f | while read file; do "$1" "$file"; done
}

注意,它不使用包含空格的文件名(见下文)。

以这个函数为例:

world(){
    sed -i 's_hello_world_g' "$1"
}

假设我想在当前目录的所有文件中将“hello”的所有实例更改为“world”。我会这样做:

toall world

为了确保文件名中的符号安全,请使用:

toall(){
    find . -type f -print0 | while IFS= read -r -d '' file; do "$1" "$file"; done
}

(但是你需要一个处理-print0的find,例如GNU find)。