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 / 问题 / 701591
Accepted
Ahmed Moselhi
Ahmed Moselhi
Asked: 2022-05-06 19:13:19 +0800 CST2022-05-06 19:13:19 +0800 CST 2022-05-06 19:13:19 +0800 CST

在文件的每一行中查找一个字符串,然后使用 bash 替换同一行中的另一个字符串

  • 772

这是我的脚本:

#! /bin/bash

# C-band Edit
dqt='"'
str5="polarization=${dqt}2${dqt}"
for x in {3600..4200} do;
    sed -i "/$str5.*$x/s/$x/$((x-600))/" satellites.xml
done

我想做的就是在satellites.xml中有str5的所有行上用3600到4200之间的x-600替换数字x,但上面的脚本给了我语法错误

linux bash
  • 2 2 个回答
  • 48 Views

2 个回答

  • Voted
  1. Best Answer
    cas
    2022-05-06T20:00:12+08:002022-05-06T20:00:12+08:00

    我突然想到你不是指文字字符串x-600,你的意思是“从 3600 到 4200 之间的每个值中减去 600”。在这种情况下,使用 perl,而不是 sed:

    echo "$str5. foo bar 3200 3964 4155 4200 4255" |
      perl -p -e "if (/$str5/) {s/3[6789]\d\d|4[01]\d\d|4200/$&-600/eg}"
    polarization="2". foo bar 3200 3364 3555 3600 4255
    

    这使用 perl 的/e正则表达式修饰符来导致右侧 ( $&-600) 被评估为 perl 表达式。 $&是一个包含匹配值的 perl 变量,因此$&-600意味着从该值中减去 600。

    如果那是您真正想要的,我会在下面留下我的原始答案。也因为它有有用的解释,这些解释仍然与上面的 perl 答案有些相关。

    与 sed 一样,perl 有一个-i就地编辑选项,因此您可以将其直接应用到您的 satellites.xml 文件中。详情请参阅man perlrun。

    perl -i -p -e "if (/$str5/) {s/3[6789]\d\d|4[01]\d\d|4200/$&-600/eg}" satellites.xml
    

    另外值得注意的是:在处理 XML 文件时,您可能应该使用 XML 解析器。幸运的是,perl 有几个可供选择,例如XML::Parser或集合libxml-perl。


    我没有您实际输入的示例,因此我制作了一个示例来演示什么会改变,什么不会:

    str5='polarization="2"'
    
    echo "$str5. foo bar 3500 3964 4155 4200 4255" | 
      sed -e "/$str5/ {s/3[6-9][0-9][0-9]/3-600/g; s/4[01][0-9][0-9]/4-600/g; s/4200/4-600/g}"
    polarization="2". foo bar 3500 3-600 4-600 4-600 4255
    

    大致翻译成英文,就是“如果当前行包含$str5( polarization="2"),则对其应用这三个s///操作”。

    笔记:

    • 无需从 3600..4200 循环。这将在一次运行中进行所有更改sed,而不是 600 次运行。

    • 您希望更改 3600-4200 的值。这意味着您需要 3 次搜索和替换操作:

      • 一个用于 3600-3999
      • 一个用于 4000-4199
      • 一个正好 4200
    • 或者,这可以通过两个s///操作来完成,将最后两个合并为一个:

        sed -e "/$str5/ { s/3[6-9][0-9][0-9]/3-600/g; s/4[01][0-9][0-9]\|4200/4-600/g }"
      
    • 可能还有许多其他方法可以优化正则表达式,但是您这样做的次数越多,将来阅读和修改它们的难度就越大。

    • 3964, 4155,4200并被更改。 3500而4255不是,它们超出了所需的范围。

    • [0-9]在某些语言环境(该范围内还有其他字符)中不会像您期望的那样工作。我不确切知道哪些语言环境,但我已经看到它经常被提及,足以知道[0-9]不能完全依赖它。如果这影响到您,您可以使用[[:digit:]]代替[0-9],或使用perl -p代替sed(因此您可以使用\d代替[0-9])。同样适用于 range [6-9],请[6789]改用。

    • 最后,要非常小心你放入$str5变量中的内容。因为它被插入到sed命令中,所以很容易破坏 sed 脚本(例如,如果$str5包含 a /,它将破坏/$str5/匹配)。sed 不知道它的脚本部分来自一个 shell 变量,它所看到的只是它被赋予运行的脚本。

      此外,整个字符串将被sed 解释为正则表达式- 这意味着正则表达式元字符不会被解释为文字字符,除非它们使用\. eg.将被解释为“任何字符”而不是文字点,除非它被转义为\.

    • 2
  2. Philippos
    2022-05-07T00:59:04+08:002022-05-07T00:59:04+08:00

    do您的尝试原则上是正确的,正如 frabjous 指出的那样,您只是错过了在分号之前关闭列表:

    for x in {3600..4200}; do
    

    但请注意,这将使您“就地”编辑文件 601 次,实际上每次执行时都会创建和删除数百个新文件。

    为避免这种情况,您可以

    1. 一次性教授sed一些对数字的基本理解,这很有趣,但不是很有用
    2. 使用可以计算数字的工具,python例如perlawk
    3. 将其减少到只有七个您可以输入的实际替代品sed:
        #! /bin/bash
        # C-band Edit
        dqt='"'
        str5="polarization=${dqt}2${dqt}"
        sed -Ei "/$str5$x/{"$(for x in {36..41}; do echo -n "s/$x([0-9]{2})/$((x-6))\\1/;"; done)";s/4200/3600/;}" satellites.xml
    

    替换将产生sed脚本

    s/36([0-9]{2})/30\1/;s/37([0-9]{2})/31\1/;s/38([0-9]{2})/32\1/;s/39([0-9]{2})/33\1/;s/40([0-9]{2})/34\1/;s/41([0-9]{2})/35\1/;s/4200/3600/
    
    • 1

相关问题

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

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

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

  • 通过标签将根文件系统传递给linux内核

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

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