AskOverflow.Dev

AskOverflow.Dev Logo AskOverflow.Dev Logo

AskOverflow.Dev Navigation

  • 主页
  • 系统&网络
  • Ubuntu
  • Unix
  • DBA
  • Computer
  • Coding
  • LangChain

Mobile menu

Close
  • 主页
  • 系统&网络
    • 最新
    • 热门
    • 标签
  • Ubuntu
    • 最新
    • 热门
    • 标签
  • Unix
    • 最新
    • 标签
  • DBA
    • 最新
    • 标签
  • Computer
    • 最新
    • 标签
  • Coding
    • 最新
    • 标签
主页 / unix / 问题 / 414042
Accepted
yukashima huksay
yukashima huksay
Asked: 2018-01-01 13:27:16 +0800 CST2018-01-01 13:27:16 +0800 CST 2018-01-01 13:27:16 +0800 CST

如何使用 grep 将输出拆分为两个文件?

  • 772

我有一个mycommand.sh无法运行两次的脚本。我想将输出拆分为两个不同的文件,一个文件包含与正则表达式匹配的行,另一个文件包含与正则表达式不匹配的行。我希望拥有的基本上是这样的:

./mycommand.sh | grep -E 'some|very*|cool[regex].here;)' --match file1.txt --not-match file2.txt

我知道我可以将输出重定向到一个文件,然后重定向到两个不同的带有和不带 -v 选项的 grep,并将它们的输出重定向到两个不同的文件。但我只是想知道是否可以用一个 grep 来做到这一点。

那么,是否有可能在一行中实现我想要的?

grep io-redirection
  • 4 4 个回答
  • 9748 Views

4 个回答

  • Voted
  1. Best Answer
    John1024
    2018-01-01T13:31:03+08:002018-01-01T13:31:03+08:00

    有很多方法可以做到这一点。

    使用 awk

    以下发送coolregex与 file1 匹配的任何行。所有其他行都转到 file2:

    ./mycommand.sh | awk '/[coolregex]/{print>"file1";next} 1' >file2
    

    这个怎么运作:

    1. /[coolregex]/{print>"file1";next}

      任何与正则表达式匹配的行都会coolregex打印到file1. 然后,我们跳过所有剩余的命令并跳转到该next行重新开始。

    2. 1

      所有其他行都发送到标准输出。 1是 awk 对 print-the-line 的神秘简写。

    也可以拆分为多个流:

    ./mycommand.sh | awk '/regex1/{print>"file1"} /regex2/{print>"file2"} /regex3/{print>"file3"}'
    

    使用进程替换

    这不像 awk 解决方案那样优雅,但为了完整起见,我们还可以结合使用多个 grep 和进程替换:

    ./mycommand.sh | tee >(grep 'coolregex' >File1) | grep -v 'coolregex' >File2
    

    我们还可以拆分为多个流:

    ./mycommand.sh | tee >(grep 'coolregex' >File1) >(grep 'otherregex' >File3) >(grep 'anotherregex' >File4) | grep -v 'coolregex' >File2
    
    • 22
  2. MiniMax
    2018-01-01T15:05:26+08:002018-01-01T15:05:26+08:00
    sed -n -e '/pattern_1/w file_1' -e '/pattern_2/w file_2' input.txt
    

    w filename- 将当前模式空间写入文件名。

    如果您希望所有匹配的行都转到file_1,所有不匹配的行都转到file_2,您可以执行以下操作:

    sed -n -e '/pattern/w file_1' -e '/pattern/!w file_2' input.txt
    

    或者

    sed -n '/pattern/!{p;d}; w file_1' input.txt > file_2
    

    解释

    1. /pattern/!{p;d};
      • /pattern/!- 否定 - 如果一行不包含pattern.
      • p- 打印当前模式空间。
      • d- 删除模式空间。开始下一个循环。
      • 因此,如果一行不包含模式,它会将这一行打印到标准输出并选择下一行。在我们的例子中,标准输出被重定向到file_2。当行与模式不匹配时,sed脚本的下一部分( ) 未到达。w file_1
    2. w file_1- 如果一行包含模式,/pattern/!{p;d};则跳过该部分(因为它仅在模式不匹配时执行),因此,该行转到file_1.
    • 8
  3. Elias
    2018-01-02T01:52:26+08:002018-01-02T01:52:26+08:00

    我喜欢这个sed解决方案,因为它不依赖 bashisms 并且以相同的方式处理输出文件。AFAIK,没有独立的 Unix 工具可以做你想做的事,所以你需要自己编程。如果我们放弃瑞士军刀方法,我们可以使用任何脚本语言(Perl、Python、NodeJS)。

    这就是在 NodeJS 中完成的方式

      #!/usr/bin/env node
    
      const fs = require('fs');
      const {stderr, stdout, argv} = process;
    
      const pattern = new RegExp(argv[2] || '');
      const yes = argv[3] ? fs.createWriteStream(argv[3]) : stdout;
      const no = argv[4] ? fs.createWriteStream(argv[4]) : stderr;
    
      const out = [no, yes];
    
      const partition = predicate => e => {
        const didMatch = Number(!!predicate(e));
        out[didMatch].write(e + '\n');
      };
    
      fs.readFileSync(process.stdin.fd)
        .toString()
        .split('\n')
        .forEach(partition(line => line.match(pattern)));
    

    示例用法

    # Using designated files
    ./mycommand.sh | partition.js pattern file1.txt file2.txt
    
    # Using standard output streams
    ./partition.js pattern > file1.txt 2> file2.txt
    
    • 0
  4. David Foerster
    2018-01-02T07:33:04+08:002018-01-02T07:33:04+08:00

    如果您不介意使用 Python 和不同的正则表达式语法:

    #!/usr/bin/env python3
    import sys, re
    
    regex, os1, os2 = sys.argv[1:]
    regex = re.compile(regex)
    with open(os1, 'w') as os1, open(os2, 'w') as os2:
        os = (os1, os2)
        for line in sys.stdin:
            end = len(line) - line.endswith('\n')
            os[regex.search(line, 0, end) is not None].write(line)
    

    用法

    ./match-split.py PATTERN FILE-MATCH FILE-NOMATCH
    

    例子

    printf '%s\n' foo bar baz | python3 match-split.py '^b' b.txt not-b.txt
    
    • 0

相关问题

  • 来自 `service | 的意外结果 grep`

  • 读取带有单词的文本文件及其出现次数和排序的打印输出

  • 命令 ls | grep 只显示目录(当它也应该显示文件时)

  • grep 什么时候计数,什么时候不计数

  • grep --line-buffered 直到 X 行?

Sidebar

Stats

  • 问题 205573
  • 回答 270741
  • 最佳答案 135370
  • 用户 68524
  • 热门
  • 回答
  • Marko Smith

    如何将 GPG 私钥和公钥导出到文件

    • 4 个回答
  • Marko Smith

    ssh 无法协商:“找不到匹配的密码”,正在拒绝 cbc

    • 4 个回答
  • Marko Smith

    我们如何运行存储在变量中的命令?

    • 5 个回答
  • Marko Smith

    如何配置 systemd-resolved 和 systemd-networkd 以使用本地 DNS 服务器来解析本地域和远程 DNS 服务器来解析远程域?

    • 3 个回答
  • Marko Smith

    如何卸载内核模块“nvidia-drm”?

    • 13 个回答
  • Marko Smith

    dist-upgrade 后 Kali Linux 中的 apt-get update 错误 [重复]

    • 2 个回答
  • Marko Smith

    如何从 systemctl 服务日志中查看最新的 x 行

    • 5 个回答
  • Marko Smith

    Nano - 跳转到文件末尾

    • 8 个回答
  • Marko Smith

    grub 错误:你需要先加载内核

    • 4 个回答
  • Marko Smith

    如何下载软件包而不是使用 apt-get 命令安装它?

    • 7 个回答
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +0800 CST
  • Martin Hope
    Wong Jia Hau ssh-add 返回:“连接代理时出错:没有这样的文件或目录” 2018-08-24 23:28:13 +0800 CST
  • Martin Hope
    Evan Carroll systemctl 状态显示:“状态:降级” 2018-06-03 18:48:17 +0800 CST
  • Martin Hope
    Tim 我们如何运行存储在变量中的命令? 2018-05-21 04:46:29 +0800 CST
  • Martin Hope
    Ankur S 为什么 /dev/null 是一个文件?为什么它的功能不作为一个简单的程序来实现? 2018-04-17 07:28:04 +0800 CST
  • Martin Hope
    user3191334 如何从 systemctl 服务日志中查看最新的 x 行 2018-02-07 00:14:16 +0800 CST
  • Martin Hope
    Marko Pacak Nano - 跳转到文件末尾 2018-02-01 01:53:03 +0800 CST
  • Martin Hope
    Kidburla 为什么真假这么大? 2018-01-26 12:14:47 +0800 CST
  • Martin Hope
    Christos Baziotis 在一个巨大的(70GB)、一行、文本文件中替换字符串 2017-12-30 06:58:33 +0800 CST
  • Martin Hope
    Bagas Sanjaya 为什么 Linux 使用 LF 作为换行符? 2017-12-20 05:48:21 +0800 CST

热门标签

linux bash debian shell-script text-processing ubuntu centos shell awk ssh

Explore

  • 主页
  • 问题
    • 最新
    • 热门
  • 标签
  • 帮助

Footer

AskOverflow.Dev

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

Language

  • Pt
  • Server
  • Unix

© 2023 AskOverflow.DEV All Rights Reserve