Heim >Backend-Entwicklung >Golang >Was entweicht in den Haufen?

Was entweicht in den Haufen?

王林
王林nach vorne
2024-02-06 10:09:071294Durchsuche

Was entweicht in den Haufen?

Frageninhalt

Ich habe diesen Code, der eigentlich überhaupt nicht zuordnen soll, aber aus irgendeinem Grund tut er das. Wie im Benchmark angegeben, erfolgen pro Vorgang zwei Zuweisungen.

Welche Zeilen der Funktion sind belegt? Warum?

Eigenschaften:

func (vi *VarInt /* int32 */) Read(input io.Reader) error {
    var (
        b     byte
        buf   = unsafe.Slice(&b, 1)
        shift int
        value uint32
    )
    for {
        _, err := io.ReadFull(input, buf)
        if err != nil {
            return err
        }

        value |= (uint32(b) & 0b01111111) << shift

        if (b & 0b10000000) == 0 {
            *vi = VarInt(value)
            return nil
        }

        shift += 7
        if shift >= 32 {
            return ErrVarIntTooLong
        }
    }
}

func (vi *VarInt /* int32 */) Write(output io.Writer) error {
    var (
        varint [5]byte
        uvalue = uint32(*vi)
        x      int
    )
    for ; ; x++ {
        vb := uint8(uvalue)

        if (vb & 0b10000000) == 0 {
            varint[x] = vb
            break
        }

        varint[x] = (vb & 0b01111111) | 0b10000000

        uvalue >>= 7
    }

    _, err := output.Write(varint[:x+1])
    if err != nil {
        return err
    }

    return nil
}

Benchmark:

func BenchmarkVarInt(b *testing.B) {
    var buf bytes.Buffer
    buf.Grow(5)
    b.ResetTimer()
    for i := 0; i < b.N; i++ {
        vi := (VarInt)(i)
        vi.Write(&buf)
        vi.Read(&buf)
        buf.Reset()
    }
}

Ich wollte buf 切片以某种方式逃逸,但我不知道如何逃逸,因为据我了解,在这种情况下切片是在堆栈上分配的结构,它将指向变量 b 作为其数据。我尝试将表达式 unsafe.Slice(&b, 1) 更改为 (*[1]byte)(unsafe.Pointer(&b))[:] aber es hat nichts geändert.


Richtige Antwort


Wenn ein Wert in einer Schnittstelle eingerahmt ist, wird er immer als escapbar betrachtet – auch wenn der Wert nie außerhalb des Aufrufstapels verwendet wird, stoppt Go an diesem Punkt das Parsen und denkt, dass jemand es erreicht haben könnte der Adresse muss der Wert auf dem Heap platziert werden.

Since Read 采用 io.ReaderWrite 采用 io.Writer,因此 buf (这是传递给这两个函数的 bytes.Buffer ) muss maskiert werden.

Auch wenn Sie dafür sorgen, dass diese Funktionen konkrete Typen annehmen bytes.Buffer (您可能不想要),但这还不够,因为 Read 调用 io.ReadFull ,它再次采用 io.Reader . Sie müssen härter arbeiten, um diesen Auftrag freizustellen.

Als Randbemerkung, für Read 中的其他问题,有一个更简单的解决方案,不需要任何 unsafe.Slice 恶作剧:只需将 var b byte 替换为 var b [1]byte (这正是 内存中相同),将b[:]传递给ReadFull,并在其他使用b的地方使用b[0].

Das obige ist der detaillierte Inhalt vonWas entweicht in den Haufen?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Dieser Artikel ist reproduziert unter:stackoverflow.com. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen