我试图使用grep来匹配包含两个不同字符串的行。我已经尝试了以下方法,但这匹配了包含不是我想要的string1或string2的行。
grep 'string1\|string2' filename
那么我如何匹配与grep只包含两个字符串的行?
我试图使用grep来匹配包含两个不同字符串的行。我已经尝试了以下方法,但这匹配了包含不是我想要的string1或string2的行。
grep 'string1\|string2' filename
那么我如何匹配与grep只包含两个字符串的行?
当前回答
我经常遇到和你一样的问题,我刚刚写了一段脚本:
function m() { # m means 'multi pattern grep'
function _usage() {
echo "usage: COMMAND [-inH] -p<pattern1> -p<pattern2> <filename>"
echo "-i : ignore case"
echo "-n : show line number"
echo "-H : show filename"
echo "-h : show header"
echo "-p : specify pattern"
}
declare -a patterns
# it is important to declare OPTIND as local
local ignorecase_flag filename linum header_flag colon result OPTIND
while getopts "iHhnp:" opt; do
case $opt in
i)
ignorecase_flag=true ;;
H)
filename="FILENAME," ;;
n)
linum="NR," ;;
p)
patterns+=( "$OPTARG" ) ;;
h)
header_flag=true ;;
\?)
_usage
return ;;
esac
done
if [[ -n $filename || -n $linum ]]; then
colon="\":\","
fi
shift $(( $OPTIND - 1 ))
if [[ $ignorecase_flag == true ]]; then
for s in "${patterns[@]}"; do
result+=" && s~/${s,,}/"
done
result=${result# && }
result="{s=tolower(\$0)} $result"
else
for s in "${patterns[@]}"; do
result="$result && /$s/"
done
result=${result# && }
fi
result+=" { print "$filename$linum$colon"\$0 }"
if [[ ! -t 0 ]]; then # pipe case
cat - | awk "${result}"
else
for f in "$@"; do
[[ $header_flag == true ]] && echo "########## $f ##########"
awk "${result}" $f
done
fi
}
用法:
echo "a b c" | m -p A
echo "a b c" | m -i -p A # a b c
你可以把它放在。bashrc中。
其他回答
当两个字符串按顺序排列时,在grep命令中放入一个模式:
$ grep -E "string1(?.*)string2" file
例如,在名为Dockerfile的文件中包含以下行:
FROM python:3.8 as build-python
FROM python:3.8-slim
要获取包含字符串的行:FROM python和as build-python,然后使用:
$ grep -E "FROM python:(?.*) as build-python" Dockerfile
然后输出将只显示包含这两个字符串的行:
FROM python:3.8 as build-python
正则表达式中的|操作符表示或。也就是说,string1或string2将匹配。你可以这样做:
grep 'string1' filename | grep 'string2'
它将把第一个命令的结果输送到第二个grep中。这应该只会给出两者都匹配的行。
搜索两个String,只突出显示string1和string2
grep -E 'string1.*string2|string2.*string1' filename | grep -E 'string1|string2'
or
grep 'string1.*string2\|string2.*string1' filename | grep -E 'string1\|string2'
你可以尝试这样做:
(pattern1.*pattern2|pattern2.*pattern1)
你的方法几乎很好,只是少了一个-w
grep -w 'string1\|string2' filename