[英]sed to replace lines matched by a pattern range if one line match a condition
我需要删除以下示例文件格式的记录:
record {
record {
my_id_1
my_name_1
}
record {
my_id_2
my_name_2
}
record {
my_id_3
my_name_3
}
}
在这种情况下,我们需要使用唯一值my_id
或my_id_2
删除所有记录数据,预期结果应该是:
record {
record {
my_id_1
my_name_1
}
record {
my_id_3
my_name_3
}
}
注意:该文件不是一个标准的谈论空格,换行符等。但核心逻辑可能是删除单词record
[包括相同单词]之间的所有内容,仅与括号中的my_id_*
字符串相关。
到目前为止,我所能做的就是写:
sed -n '/record/{:a;N;/}/!ba; /my_id_2/p}' file.conf
事实上,它找到了我需要的东西,但我无法删除它我只打印我想要清除的行。
sed
是我的主要选择,但python
regex
可能很好,所以我可以将它传递给 ansible。
##########################
这也是我的 PYTHON 尝试
##########################
使用 python 这是我得到的最接近的:
第一次尝试: (?=my_id_2)([^}]+)(?=\})
匹配结果:
`my_id_2 my_name_2`
然后稍作修改: (?=record )([^}]+)(?=\})
匹配结果:
Match 1
`1. record { record { my_id_1 my_name_1`
Match 2
`1. record { my_id_2 my_name_2`
Match 3
`1. record { my_id_3 my_name_3`
谢谢。
你可以试试:
$ sed -i.bak '/record/{:a;N;/}/!ba; /my_id_2/d}' file.conf
$ cat file.conf
record {
record {
my_id_1
my_name_1
}
record {
my_id_3
my_name_3
}
}
以上内容就地修改了文件并创建了文件的备份,名为file.conf.bak
。 如果您不需要原始文件的备份,您可以从-i.bak
中删除.bak
。
相信这可以满足您的要求,但它会在前面加上一个空行。 它只是将它们作为段落括起来,然后删除具有该记录名称的段落。
sed '/./{H;$!d} ;x; /my_id_2/d' file.conf
你会认为你可以简单地用额外的; 1d
; 1d
附加到您的 sed 语句。
sed '/./{H;$!d} ;x; /my_id_2/d ; 1d' file.conf
但是,这没有任何效果,并且第 1 行仍然是空白的。 您可以通过将其传递给后续的sed
命令来克服这个问题,但这似乎很不稳定。 也许其他人知道为什么。
sed '/./{H;$!d} ;x; /my_id_2/d' file.conf | sed '1d'
这可能对您有用(GNU sed):
sed -E '/record \{/{:a;N;/\}/!ba;/my_id_2/!b;s/(record \{.*)\1.*/\1/;t;d}' file
使用您的原始方法,但使用贪婪进行额外测试,如果成功,则保留匹配字符串的前面。 如果没有这样的匹配,恢复并像以前一样删除整个匹配的字符串。
注意这可能会引入一些不需要的填充,将其删除作为练习留给读者;-)
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.