我试图使用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中。

其他回答

正如人们建议的perl和python,以及复杂的shell脚本,这里有一个简单的awk方法:

awk '/string1/ && /string2/' filename

看了评论,得到了公认的答案:不,这没有多行;但这也不是问题作者想要的。

如果您有一个grep,其中有一个-P选项用于有限的perl regex,您可以使用

grep -P '(?=.*string1)(?=.*string2)'

它的优点是处理重叠的字符串。使用perl作为grep更直接,因为你可以更直接地指定and逻辑:

perl -ne 'print if /string1/ && /string2/'
grep ‘string1\|string2’ FILENAME 

GNU grep 3.1版

你的方法几乎很好,只是少了一个-w

grep -w 'string1\|string2' filename

当两个字符串按顺序排列时,在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