-3

我遇到了 golang 1.16 的奇怪行为。对于我的一个项目,我需要遍历字符串中的符文。所以,我创建了一个简单的迭代器,如下所示:

func iterRunes(s string) <-chan rune {
    runes := make(chan rune);
    go func() {
        defer close(runes);
        for _, char := range s {
            runes <- char;
        }
    } ()
    return runes;
}

它工作得很好,但我还需要枚举由此产生的值。所以,我写了另一个看起来像:

func enumRunes(runes <-chan rune) <-chan struct {int; rune} {
    eRunes := make(chan struct {int; rune});
    go func() {
        defer close(eRunes);
        i := 0;
        for r := range runes {
            eRunes <- struct {int; rune} {i, r};
            i++;
        }
    } ()
    return eRunes;
}

它也可以完美运行。但是如果我尝试将它们组合在一起enumRunes(iterRunes(pattern)),我会遇到问题。我有两个代码片段要显示。

第一个是:

    tmp := enumRunes(iterRunes(pattern))
    z := <-tmp;
    for z.int < utf8.RuneCountInString(pattern) {
        fmt.Println("z = <- tmp;")
        fmt.Println(z)
        z = <- tmp;
    }

它的输出是这样的:

...
z = <- tmp;
{0 0}
{0 0}
z = <- tmp;
{0 0}
{0 0}
z = <- tmp;
{0 0}
{0 0}
z = <- tmp;
...

z是结构类型。所以,我进入了一个无限循环,因为结构由于某种原因没有更新。第二个片段:

    tmp := iterRunes(pattern)
    z := <-tmp;
    for qq := 0; qq < utf8.RuneCountInString(pattern); qq++ {
        fmt.Println("z = <- tmp;")
        fmt.Println(z)
        z = <- tmp;
    }

z是符文类型,它工作正常。问题是为什么z第一个片段没有更新。先感谢您。

4

1 回答 1

2

我希望你知道(参考

来自关闭通道的接收立即返回零值

有了这个,让我们看看你的问题!

z.int < utf8.RuneCountInString(pattern)

只要 z.int 小于字符串的符文计数,即使该值是默认值 0,这始终会得到满足。

有了这个上下文,您现在可以看到,即使通道已关闭,由于您正在从中读取,它也会返回一个空的{0, 0} struct. 哪个集合z.int to 0并且循环继续无穷大。

于 2021-03-20T20:51:20.080 回答