Go entschlüsseln: leere Struktur

PHPz
Freigeben: 2024-09-11 12:30:32
Original
850 Leute haben es durchsucht

Decrypt Go: empty struct

In Go belegt eine normale Struktur normalerweise einen Speicherblock. Es gibt jedoch einen Sonderfall: Wenn es sich um eine leere Struktur handelt, ist ihre Größe Null. Wie ist das möglich und welchen Nutzen hat eine leere Struktur?

Dieser Artikel wird zuerst im mittleren MPP-Plan veröffentlicht. Wenn Sie ein mittlerer Benutzer sind, folgen Sie mir bitte auf mittlerem Niveau. Vielen Dank.

type Test struct {
    A int
    B string
}

func main() {
    fmt.Println(unsafe.Sizeof(new(Test)))
    fmt.Println(unsafe.Sizeof(struct{}{}))
}

/*
8
0
*/
Nach dem Login kopieren

Das Geheimnis der leeren Struktur

Spezielle Variable: Nullbasis

Eine leere Struktur ist eine Struktur ohne Speichergröße. Diese Aussage ist richtig, aber genauer gesagt hat sie tatsächlich einen besonderen Ausgangspunkt: die Nullbasisvariable. Dies ist eine globale uintptr-Variable, die 8 Bytes belegt. Immer wenn unzählige struct {}-Variablen definiert werden, weist der Compiler die Adresse dieser Zerobase-Variablen zu. Mit anderen Worten: In Go verwendet jede Speicherzuweisung mit der Größe 0 dieselbe Adresse, &zerobase.

Beispiel

package main

import "fmt"

type emptyStruct struct {}

func main() {
    a := struct{}{}
    b := struct{}{}
    c := emptyStruct{}

    fmt.Printf("%p\n", &a)
    fmt.Printf("%p\n", &b)
    fmt.Printf("%p\n", &c)
}

// 0x58e360
// 0x58e360
// 0x58e360
Nach dem Login kopieren

Die Speicheradressen der Variablen einer leeren Struktur sind alle gleich. Dies liegt daran, dass der Compiler während der Kompilierung &zerobase zuweist, wenn er auf diese spezielle Art der Speicherzuweisung stößt. Diese Logik befindet sich in der mallocgc-Funktion:

//go:linkname mallocgc  
func mallocgc(size uintptr, typ *_type, needzero bool) unsafe.Pointer {  
    ...
    if size == 0 {  
       return unsafe.Pointer(&zerobase)  
    }
    ...
Nach dem Login kopieren

Das ist das Geheimnis der leeren Struktur. Mit dieser speziellen Variablen können wir viele Funktionalitäten realisieren.

Leere Struktur und Speicherausrichtung

Wenn eine leere Struktur Teil einer größeren Struktur ist, belegt sie normalerweise keinen Speicher. Es gibt jedoch einen Sonderfall, wenn die leere Struktur das letzte Feld ist; es löst eine Gedächtnisausrichtung aus.

Beispiel

type A struct {
    x int
    y string
    z struct{}
}
type B struct {
    x int
    z struct{}
    y string
}

func main() {
    println(unsafe.Alignof(A{}))
    println(unsafe.Alignof(B{}))
    println(unsafe.Sizeof(A{}))
    println(unsafe.Sizeof(B{}))
}

/**
8
8
32
24
**/
Nach dem Login kopieren

Wenn ein Zeiger auf ein Feld vorhanden ist, liegt die zurückgegebene Adresse möglicherweise außerhalb der Struktur, was möglicherweise zu Speicherverlusten führt, wenn der Speicher beim Freigeben der Struktur nicht freigegeben wird. Wenn eine leere Struktur das letzte Feld einer anderen Struktur ist, wird daher aus Sicherheitsgründen zusätzlicher Speicher zugewiesen. Befindet sich die leere Struktur am Anfang oder in der Mitte, ist ihre Adresse dieselbe wie die der nächsten Variablen.

type A struct {  
    x int  
    y string  
    z struct{}  
}  
type B struct {  
    x int  
    z struct{}  
    y string  
}  

func main() {  
    a := A{}  
    b := B{}  
    fmt.Printf("%p\n", &a.y)  
    fmt.Printf("%p\n", &a.z)  
    fmt.Printf("%p\n", &b.y)  
    fmt.Printf("%p\n", &b.z)  
}

/**
0x1400012c008
0x1400012c018
0x1400012e008
0x1400012e008
**/
Nach dem Login kopieren

Anwendungsfälle der leeren Struktur

Der Hauptgrund für die Existenz der leeren Struktur struct{} besteht darin, Speicher zu sparen. Wenn Sie eine Struktur benötigen, sich aber nicht um deren Inhalt kümmern, sollten Sie die Verwendung einer leeren Struktur in Betracht ziehen. Die wichtigsten zusammengesetzten Strukturen von Go wie Map, Chan und Slice können alle struct{} verwenden.

Karte & Struktur{}

// Create map
m := make(map[int]struct{})
// Assign value
m[1] = struct{}{}
// Check if key exists
_, ok := m[1]
Nach dem Login kopieren

chan & struct{}

Ein klassisches Szenario kombiniert Kanal und Struktur{}, wobei Struktur{} oft als Signal verwendet wird, ohne sich um seinen Inhalt zu kümmern. Wie in früheren Artikeln analysiert, besteht die wesentliche Datenstruktur eines Kanals aus einer Verwaltungsstruktur und einem Ringpuffer. Der Ringpuffer wird mit Null belegt, wenn struct{} als Element verwendet wird.

Die einzige gemeinsame Verwendung von chan und struct{} dient der Signalübertragung, da die leere Struktur selbst keinen Wert tragen kann. Im Allgemeinen wird es ohne Pufferkanäle verwendet.

// Create a signal channel
waitc := make(chan struct{})

// ...
goroutine 1:
    // Send signal: push element
    waitc <- struct{}{}
    // Send signal: close
    close(waitc)

goroutine 2:
    select {
    // Receive signal and perform corresponding actions
    case <-waitc:
    }    
Nach dem Login kopieren

Ist struct{} in diesem Szenario unbedingt erforderlich? Nicht wirklich, und der eingesparte Speicher ist vernachlässigbar. Der entscheidende Punkt ist, dass der Elementwert von chan nicht berücksichtigt wird, daher wird struct{} verwendet.

Zusammenfassung

  1. Eine leere Struktur ist immer noch eine Struktur, nur mit einer Größe von 0.
  2. Alle leeren Strukturen haben dieselbe Adresse: die Adresse von Zerobase.
  3. Wir können die nicht speicherbelegende Funktion der leeren Struktur nutzen, um Code zu optimieren, z. B. durch die Verwendung von Karten zum Implementieren von Mengen und Kanälen.

Referenzen

  1. Die leere Struktur, Dave Cheney
  2. Gehen Sie zu 最细节篇— struct{} 空结构体究竟是啥?

Das obige ist der detaillierte Inhalt vonGo entschlüsseln: leere Struktur. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Quelle:dev.to
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage