Answers:
我实现了一个脚本,正是这样做的。
if [ $# -eq 0 ]; then
PATTERNS=(NAME AUTHOR EXAMPLES FILES)
else
PATTERNS=( "$@" )
fi
[ ${#PATTERNS[@]} -lt 1 ] && echo "Needs at least 1 pattern to search for" && exit 1
for i in $(find /usr/share/man/ -type f); do
TMPOUT=$(zgrep -l "${PATTERNS[0]}" "$i")
[ -z "$TMPOUT" ] && continue
for c in `seq 1 $((${#PATTERNS[@]}-1))`; do
TMPOUT=$(echo "$TMPOUT" | xargs zgrep -l "${PATTERNS[$c]}")
[ -z "$TMPOUT" ] && break
done
if [ ! -z "$TMPOUT" ]; then
#echo "$TMPOUT" # Prints the whole path
MANNAME="$(basename "$TMPOUT")"
man "${MANNAME%%.*}"
fi
done
猜猜那是浪费时间:(
编辑:好像
man -K expr1 expr2 expr3
没有工作吗?
编辑:您现在可以通过通过搜索条件传递脚本 ./script foo bar
or
,我只是以为是and
因为我没有正确测试它。
关于编写脚本的几点思考:
使用manpath
获得的手册页的位置(一个或多个)。如果我添加/home/graeme/.cabal/bin
到我的PATH
,manpath
(和man
)会发现在手册页/home/graeme/.cabal/share/man
。
在搜索之前,使用man本身对页面进行解压缩和格式化,这样您就可以搜索man文本本身,而无需搜索原始文件中的任何注释等。使用man
将潜在地处理多种格式。
将格式化的页面保存在临时文件中将避免多次解压缩,并应大大加快速度。
这就是(bash
和GNU查找):
#!/bin/bash
set -f; IFS=:
trap 'rm -f "$temp"' EXIT
temp=$(mktemp --tmpdir search_man.XXXXXXXXXX)
while IFS= read -rd '' file; do
man "$file" >"$temp" 2>/dev/null
unset fail
for arg; do
if ! grep -Fq -- "$arg" "$temp"; then
fail=true
break
fi
done
if [ -z "$fail" ]; then
file=${file##*/}
printf '%s\n' "${file%.gz}"
fi
done < <(find $(manpath) -type d ! -name 'man*' -prune -o -type f -print0)
不如@polym的答案完整,但我将建议类似
while IFS= read -rd $'\0' f; do
zgrep -qwm1 'foo' "$f" && \
zgrep -qwm1 'bar' "$f" && \
zgrep -qwm1 'baz' "$f" && \
printf '%s\n' "$f"
done < <(find /usr/share/man -name '*.gz' -print0)
请注意,我在-上添加了一个-w
(单词匹配)开关,greps
这可能不是您想要的(要包括foo lish和nut bar这样的匹配吗?)
这种方法未经测试,但相当简单(愚蠢的简单方法),我希望它可以工作,即使效率不高:
#!/bin/bash
if [ "$#" -eq 0 ]; then
echo "Provide arguments to search all man pages for all arguments." >&2
echo "Putting rare search terms first will improve performance." >&2
exit
fi
if [ "$#" -eq 1 ]; then
exec man -K "$@"
fi
pages=( $(man -wK "$1") )
shift
while [ "$#" -gt 1 ]; do
pages=( $(zgrep -l "$1" "${pages[@]}") )
shift
done
exec man "${pages[@]}"