【问题标题】:How to find non-matching strings with Regex in Sublime?如何在 Sublime 中使用正则表达式查找不匹配的字符串?
【发布时间】:2025-12-12 06:35:02
【问题描述】:

我有这段文字:

    <Path Fill="None"
        PathData="M244.87,363.97 L245.38,363.91 M245.38,363.91 L245.46,363.84 M245.46,363.84 L245.52,363.75 M245.52,363.75 L245.54,363.7 M245.54,363.7 L246.07,370.18 M246.07,370.18 L245.95,370.25 M245.95,370.25 L245.8,370.37 M245.8,370.37 L245.63,370.54 M245.63,370.54 L245.52,370.73 M245.52,370.73 L245.42,370.9 M245.42,370.9 L245.17,368.03 M245.17,368.03 L244.87,363.97"
        Stroke="#898989" StrokeWidth="0.5"/>
    <Path Fill="None"
        PathData="M247.4,371.21 L247.49,371.16 M247.49,371.16 L247.91,371.13 M247.91,371.13 L249.74,371.01 M249.74,371.01 L252.52,370.82 M252.52,370.82 L252.72,370.83 M252.72,370.83 L252.72,370.84 M252.72,370.84 L252.71,370.89 M252.71,370.89 L252.72,370.95 M252.72,370.95 L252.75,371.38 M252.75,371.38 L251.86,371.44 M251.86,371.44 L249.62,371.63 M249.62,371.63 L247.55,371.79 M247.55,371.79 L247.51,371.35 M247.51,371.35 L247.47,371.28 M247.47,371.28 L247.42,371.22 M247.42,371.22 L247.4,371.21"
        Stroke="#878787" StrokeWidth="0.5"/>
    <Path Fill="None"
        PathData="M246.46,372.67 L246.47,372.05 M246.47,372.05 L246.47,372.05 M246.47,372.05 L246.52,372.07 M246.52,372.07 L246.58,372.09 M246.58,372.09 L247.44,372.02 M247.44,372.02 L248.68,371.91 M248.68,371.91 L248.81,373 M248.81,373 L248.07,373.06 M248.07,373.06 L247.88,373.07 M247.88,373.07 L248.54,379.11 M248.54,379.11 L247.62,379.18 M247.62,379.18 L247.2,379.21 M247.2,379.21 L247.15,379.24 M247.15,379.24 L247.12,379.27 M247.12,379.27 L247.06,379.17 M247.06,379.17 L246.83,376.84 M246.83,376.84 L246.46,372.67"
        Stroke="#898989" StrokeWidth="0.5"/>

我正在尝试查找并删除不是某种颜色的路径,即 - #898989。我想使用正则表达式来查找不匹配的字符串。

我正在尝试以下方法:

.*(<Path Fill).*(\r\n|\r|\n).*(\r\n|\r|\n).*(?!#898989).*(\r\n|\r|\n)

但这与我用来查找匹配字符串的返回相同:

.*(<Path Fill).*(\r\n|\r|\n).*(\r\n|\r|\n).*(#898989).*(\r\n|\r|\n)

我认为?! 是一个否定的前瞻,并且会排除这些字符串。不过,它似乎并没有改变结果。

有什么帮助吗?

【问题讨论】:

  • 试试.*(?:&lt;Path Fill).*[\r\n].*[\r\n].*Stroke=\"(?!#898989).*[\r\n]
  • 这似乎也有效 - &lt;Path Fill((?!#878787).)*?/&gt;请在单行模式下尝试此正则表达式。 测试链接:regex101.com/r/Cuvxos/1
  • @AKSingh 你为什么不写下你的第一条评论作为答案呢? OP 不能做单行模式,因为 SVG 路径覆盖了几行,他们想选择整个东西以便摆脱它。
  • @MattDMo 我会在几个小时后。到那时,OP 还会提到他在我的解决方案中面临的任何其他问题(如果有的话)。
  • @AKSingh - 谢谢!这似乎奏效了。但是……你有解释为什么吗?我很困惑为什么我的不起作用...

标签: regex sublimetext


【解决方案1】:

您的问题有很多正则表达式解决方案。让我们首先讨论一下为什么您提出的正则表达式不能按预期工作。


问题

.*(<Path Fill).*(\r\n|\r|\n).*(\r\n|\r|\n).*(?!#898989).*(\r\n|\r|\n)

问题出现在零件上

.*(?!#898989).*(\r\n|\r|\n)

正则表达式只是说尽可能多地匹配任何东西。匹配后,检查当前位置是否没有#898989。然后又...

尽可能多地匹配任何东西导致了问题。 第一个.*实际上是捕获整行

        Stroke="#898989" StrokeWidth="0.5"/>

然后(?!#898989) 发挥作用,这将成功,因为在&gt; 之后没有#898989。为了清楚起见,将正则表达式更改为 -

.*(?:<Path Fill).*[\r\n].*[\r\n](.*)(?!#898989).*

这个正则表达式做同样的事情。在这个正则表达式中,(\r\n|\n|\r) 被替换为 [\r\n]。起始括号(?:&lt;Path Fill) 没有捕获任何内容。但是,这次#898989 之前的.*(...) 包围,以突出显示被它捕获的文本

观察黄线以查看.*#898989 之前捕获的内容。这是链接:https://regex101.com/r/2R54uW/1


更正

正如 cmets 中已经提到的,正则表达式可以通过强制 .* 停止在 Stroke=" 然后进行位置检查来纠正。

.*(?:<Path Fill).*[\r\n].*[\r\n].*Stroke=\"(?!#898989).*[\r\n]

这是另一个做同样事情的正则表达式 -

.*(?:<Path Fill).*[\r\n].*[\r\n]((?!#878787).)*/>

最后的想法

尝试使用[\r\n] 代替(\r\n|\r|\n),因为字符类比交替更快

如果您有任何其他疑问,请发表评论。

【讨论】:

    【解决方案2】:

    如果路径中不能有&lt;&gt; 字符,您可以断言颜色在关闭/&gt; 之前不会出现

    <Path Fill(?![^<>]*#898989[^<>]*/>)[^<>]*/>
    

    Regex demo

    如果它应该是专门的 Stroke

    <Path Fill(?![^<>]*Stroke="#898989"[^<>]*/>)[^<>]*/>
    

    Regex demo

    【讨论】: