Rumah >pembangunan bahagian belakang >Golang >Apa yang terlepas ke dalam timbunan?

Apa yang terlepas ke dalam timbunan?

王林
王林ke hadapan
2024-02-06 10:09:071304semak imbas

Apa yang terlepas ke dalam timbunan?

Kandungan soalan

Saya mempunyai kod ini yang sepatutnya tidak memperuntukkan sama sekali, tetapi atas sebab tertentu ia berlaku. Seperti yang dikatakan penanda aras, 2 peruntukan berlaku setiap operasi.

Barisan fungsi manakah yang diperuntukkan? kenapa?

Ciri-ciri:

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
}

Tanda aras:

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()
    }
}

Saya mahu buf 切片以某种方式逃逸,但我不知道如何逃逸,因为据我了解,在这种情况下切片是在堆栈上分配的结构,它将指向变量 b 作为其数据。我尝试将表达式 unsafe.Slice(&b, 1) 更改为 (*[1]byte)(unsafe.Pointer(&b))[:] tetapi ia tidak mengubah apa-apa.


Jawapan Betul


Apabila nilai dikotak-kotak dalam antara muka, ia sentiasa dianggap boleh dilepaskan - walaupun nilai itu tidak pernah digunakan di luar timbunan panggilan, Go akan berhenti menghuraikan pada ketika itu Dan berfikir bahawa seseorang mungkin telah memegang daripada alamat, nilai mesti diletakkan pada timbunan.

Sejak Read 采用 io.ReaderWrite 采用 io.Writer,因此 buf (这是传递给这两个函数的 bytes.Buffer ) mesti terlepas.

Walaupun anda membuat fungsi ini mengambil jenis konkrit bytes.Buffer (您可能不想要),但这还不够,因为 Read 调用 io.ReadFull ,它再次采用 io.Reader . Anda perlu bekerja lebih keras daripada itu untuk mendapatkan tugasan ini dikecualikan.

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

Atas ialah kandungan terperinci Apa yang terlepas ke dalam timbunan?. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Kenyataan:
Artikel ini dikembalikan pada:stackoverflow.com. Jika ada pelanggaran, sila hubungi admin@php.cn Padam