登录
首页 >  Golang >  Go问答

使用正则表达式在Go中拆分输入字符串

来源:stackoverflow

时间:2024-03-21 08:09:32 186浏览 收藏

在 Go 中,使用正则表达式拆分输入字符串时,可以采用以下方法: 1. 避免用引号分割: - 使用正则表达式 ^"|['"](\w+(?:\.\w+)*)['"]|(\w+),该表达式匹配字符串中的非引号文本和引号内的文本。 - 该正则表达式使用“垃圾桶技巧”,将真实数据存储在捕获组中,即使存在嵌套引号,也能正常工作。 2. 匹配周围文本并进行捕获: - 使用正则表达式 (\'[^.'"]+(?:\.[^.'"]+)+\')|(\"[^.'"]+(?:\.[^. '"]+)+\")|(?:([^.'"]+)\.?)|(?:\.([^.'\"]+)),该表达式匹配四种场景:单引号文本、双引号文本、非引号文本和 . 前面的文本。 - 该正则表达式通过捕获组获取非引号文本和引号内的文本。

问题内容

如何在 go 中通过正则表达式分割下面的输入字符串?字符串示例:

我知道如何按点分割,但如何避免用引号分割?

"a.b.c.d" -> ["a", "b", "c", "d"]
"a."b.c".d" -> ["a", "b.c", "d"]
"a.'b.c'.d" -> ["a", "b.c", "d"]

解决方案


这是另一个选项,其正则表达式稍微不那么麻烦。它使用 trash bin trick。因此,真实数据位于(第一和第二)捕获组。

它甚至可以使用这样的嵌套引号:"a.'b.c'.d.e."f.g.h"",只要不存在 2 层或更多级别的递归(如下所示:"a. 'b."c.d"'",引号内引号内引号)。

正则表达式是这样的:^"|['"](\w+(?:\.\w+)*)['"]|(\w+)

代码:

package main

import (
    "regexp"
    "fmt"
)

func main() {
    var re = regexp.mustcompile(`^"|['"](\w+(?:\.\w+)*)['"]|(\w+)`)
    var str = `"a.'b.c'.d.e."f.g.h""`

    result := re.findallstringsubmatch(str, -1)
    for _, m := range result {
        if (m[1] != "" || m[2] != "") {
            fmt.print(m[1] + m[2] + "\n")
        }
    }
}

输入:

"a.'b.c'.d.e."f.g.h""

输出:

a
b.c
d
e
f.g.h

由于 go 不支持负向先行,我认为找到与您想要拆分的 . 匹配的正则表达式并不容易/可能。相反,您可以匹配周围的文本并仅进行适当的捕获:

所以正则表达式本身有点难看,但这里是细分(忽略 go 的转义字符):

(\'[^.'"]+(?:\.[^.'"]+)+\')|(\"[^.'"]+(?:\.[^. '"]+)+\")|(?:([^.'"]+)\.?)|(?:\.([^.'\"]+))

此正则表达式匹配四种场景,并捕获这些匹配的某些子集:

  • (\'[^.'"]+(?:\.[^.'"]+)+\') - 匹配并捕获单引号文本
    • \' - 按字面意思匹配 '
    • [^.'"]+ - 匹配非引号和非句点的序列
    • (?:\.[^.'"]+)+ - 匹配句点后跟一系列非引号和非句点,根据需要重复多次。未捕获。
    • \' - 按字面意思匹配 '
  • (\"[^.'"]+(?:\.[^.'"]+)+\") - 匹配并捕获双引号文本
    • 与上面相同,但加双引号
  • (?:([^.'"]+)\.?) - 匹配由可选 . 进行的文本,不捕获 .
    • ([^.'"]+) - 匹配并捕获非引号和非句点的序列
    • \.? - 可选择匹配句点(可选择捕获分隔文本的最后一位)
  • (?:\.([^.'"]+)) - 匹配 . 前面的文本,不捕获 .
    • 与上面相同,但句点位于捕获组之前,并且也是非可选的

转储捕获的示例代码:

package main

import (
    "fmt"
    "regexp"
)

func main() {
    re := regexp.MustCompile("('[^.'\"]+(?:\\.[^.'\"]+)+')|(\"[^.'\"]+(?:\\.[^.'\"]+)+\")|(?:([^.'\"]+)\\.?)|(?:\\.([^.'\"]+))")
    txt := "a.b.c.'d.e'"

    result:= re.FindAllStringSubmatch(txt, -1)

    for k, v := range result {
        fmt.Printf("%d. %s\n", k, v)
    }
}

今天关于《使用正则表达式在Go中拆分输入字符串》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>