使用 Bash 中的另一组行过滤行

Filter lines with the other set of lines in Bash

我的标准输入有线条

$printf "C\nB\nA\n"
C
B
A

并且我想过滤掉出现在其他一些标准输入上的行(或子字符串或正则表达式 - 更简单的):

$printf "B\nA\n"
B
A

我预计只有 C 条目被过滤。

我试过

$printf "C\nB\nA\n" | grep -v `printf "B\nA\n"`

但后来我得到了

grep: A: No such file or directory

如何根据其他命令返回的行对标准输入进行过滤?

您可以使用 grep-f 选项:

Matching Control
    -f FILE, --file=FILE
          Obtain patterns from FILE, one per line.
          [...]

并使用 <(command) 语法将命令的输出用作要使用的内容:

$ printf "C\nB\nA\nA\nC\nB" | grep -vf <(printf "A\nB\n")
C
C

使用 awk 控制每行之间的匹配方式:

$ printf "C\nB\nA\n" | awk 'NF == FNR { a[[=10=]] = 1; next } a[[=10=]]' \
  <(printf "A\nB\n") - 

通过改变 a[[=13=]] 你可以定义每个应该如何匹配,即打印行 来自 file1.txt 第一列在 file2.txt:

$ awk 'NF == FNR { a[[=11=]] = 1; next } a[]' file2.txt file1.txt
#                                      ^ Print if column 1 from file1.txt
#                                        is in file2.txt

打印来自 file1.txt 的行,这些行包含在来自 file2.txt 的第一列中:

$ awk 'NF == FNR { a[] = 1; next } a[[=12=]]' file2.txt file1.txt
#                                      ^ Print if line from file1.txt match
#                    ^ Store column one from file2.txt

你可以用

printf "C\nB\nA\n" | grep -v C

随后

printf "C\nB\nA\n" | grep -v C | grep -v B

然后

printf "C\nB\nA\n" | grep -v A | grep -v B

如果你想从文件中进行 假设文件包含

C
B
A

然后:

cat file | grep -v B | grep -v A

将打印

C

如果你只想通过管道过滤一次

cat file | grep -v 'B\|A'
C