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 / 问题 / 776271
Accepted
dhm
dhm
Asked: 2024-05-11 18:16:48 +0800 CST2024-05-11 18:16:48 +0800 CST 2024-05-11 18:16:48 +0800 CST

如何使用 sed 替换最后出现的 n 个字符串

  • 772

我可以将第三次出现的字符串替换为:

sed 's/OLD/NEW/3'

或者,对于 的 GNU 实现sed,从第三个开始的所有出现:

sed 's/OLD/NEW/3g'

但是如何替换最后 3 个(或 n 个)出现的位置呢?

bash
  • 4 4 个回答
  • 191 Views

4 个回答

  • Voted
  1. Kusalananda
    2024-05-11T18:33:35+08:002024-05-11T18:33:35+08:00

    更改与特定模式匹配的字符串的最后三个实例的最简单方法是反转每行输入并替换反转的字符串:

    $ echo 'this OLD is OLD OLD OLD!' | rev | sed -e 's/DLO/WEN/' -e 's/DLO/WEN/' -e 's/DLO/WEN/' | rev
    this OLD is NEW NEW NEW!
    

    如果已知相关行的具体格式,可能会有更好的方法来执行此操作。例如,在我上面使用的测试字符串中,我们可以简单地替换OLD OLD OLD为NEW NEW NEW.

    还要注意,由于正则表达式是从左到右方向应用的,因此替换反转的字符串可能需要与原始表达式的简单反转不同的模式。例如,考虑替换 中的前两个字母abab,当反转时必须替换中的最后两个字母baba。


    sed反转一行可以这样完成,使用@作为枢轴字符将字符串分为未反转和反转的部分:

    :loop
    s/^\([^@]\)\([^@]*\)@\{0,1\}\(.*\)/\2@\1\3/
    t loop
    s/@//
    
    • 4
  2. Stéphane Chazelas
    2024-05-11T22:10:47+08:002024-05-11T22:10:47+08:00

    一种方法可能是:

    sed '
      # replace *all* OLDs with newline:
      s/OLD/\
    /g
      :1
        # replace one newline back to a OLD as long as it is
        # followed by at least 3 newlines, relying on the fact
        # that the first .* will match greedily as much as possible.
        s/\(.*\)\n\(.*\n.*\n.*\n\)/\1OLD\2/
    
        # repeat as long as the above was successful:
      t1
      # replace the remaining newlines with NEW:
      s/\n/NEW/g'
    

    使用 时perl,进行两次替换,一次不做任何更改以记录出现的次数,第二次仅替换第 (n-2)到第 n次出现:

    perl -pe '$n = s/OLD/$&/g; s/OLD/--$n < 3 ? "NEW" : $&/ge'
    

    对于任意字符串和最大出现次数:

    perl -spe '$n = s/\Q$old\E/$&/g;
               s/\Q$old\E/--$n < $max ? $new : $&/ge
              ' -- -old=OLD -new=NEW -max=3
    

    s/OLD/$&/g将所有出现的内容替换为自身(包含's$&等匹配内容)并返回替换数。或者,您可以使用which 将后面的Nothing替换为Nothing,用于标记要替换的内容的开始。sed&s/OLD\K//gOLD\K

    • 2
  3. Ed Morton
    2024-05-13T21:16:55+08:002024-05-13T21:16:55+08:00

    使用 GNU awk 来patsplit():

    $ echo 'OLDOLDOLDOLDOLD' |
        awk '
            {
                nf = patsplit($0,f,"OLD",s)
                out = s[0]
                for ( i=1; i<=nf; i++ ) {
                    out = out ( i>(nf-3) ? "NEW" : f[i] ) s[i]
                }
                print out
            }
        '
    OLDOLDNEWNEWNEW
    

    while (match())您可以在任何带有orwhile(index())和s 循环的 awk 中执行相同的操作substr(),只是需要编写更多代码。上面的内容被OLD视为正则表达式,如果这不是您想要的,则使用循环while(index())或编写代码来转义OLD可能包含的任何元字符,例如在使用文字字符串的任何 awk 中:

    echo 'OLDOLDOLDOLDOLD' |
        awk -v old="OLD" -v new="NEW" -v cnt=3 '
            {
                nf = strsplit($0,f,old,s)
                out = s[0]
                for ( i=1; i<=nf; i++ ) {
                    out = out ( i>(nf-cnt) ? "NEW" : f[i] ) s[i]
                }
                print out
            }
    
            function strsplit(str,flds,tgt,seps,    pos,cnt,lgth)    {
                lgth = length(tgt)
                while ( pos = index(str,tgt) ) {
                    seps[cnt++] = substr(str,1,pos-1)
                    flds[cnt]   = substr(str,pos,lgth)
                    str         = substr(str,pos+lgth)
                }
                seps[cnt] = str
                return cnt+0
            }
        '
    OLDOLDNEWNEWNEW
    
    • 1
  4. Best Answer
    user9101329
    2024-05-15T15:31:33+08:002024-05-15T15:31:33+08:00

    虽然此操作没有直接的 sed 选项,但这里有一个实用的方法:将最后一次出现的字符串替换为:

    $ sed -E 's/(.*)OLD/\1NEW/' input_file
    

    由于 n 通常很小,您可以简单地将其传递到同一个命令 n 次:

    $ sed -E 's/(.*)OLD/\1NEW/' input_file | sed -E 's/(.*)OLD/\1NEW/' | sed -E 's/(.*)OLD/\1NEW/'
    

    或者更简单地重复此命令 n 次:

    $ sed -Ei 's/(.*)OLD/\1NEW/' input_file
    

    (请注意,在这种情况下 input_file 会发生变化)

    • 1

相关问题

  • 通过命令的标准输出以编程方式导出环境变量[重复]

  • 从文本文件传递变量的奇怪问题

  • 虽然行读取保持转义空间?

  • `tee` 和 `bash` 进程替换顺序

  • 运行一个非常慢的脚本直到它成功

Sidebar

Stats

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

    模块 i915 可能缺少固件 /lib/firmware/i915/*

    • 3 个回答
  • Marko Smith

    无法获取 jessie backports 存储库

    • 4 个回答
  • Marko Smith

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

    • 4 个回答
  • Marko Smith

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

    • 5 个回答
  • Marko Smith

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

    • 3 个回答
  • 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
    user12345 无法获取 jessie backports 存储库 2019-03-27 04:39:28 +0800 CST
  • Martin Hope
    Carl 为什么大多数 systemd 示例都包含 WantedBy=multi-user.target? 2019-03-15 11:49:25 +0800 CST
  • Martin Hope
    rocky 如何将 GPG 私钥和公钥导出到文件 2018-11-16 05:36:15 +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

热门标签

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