Zum Inhalt springen

Go-Frontend-Bridge

Wails bietet eine direkte, im Speicher befindliche Brücke zwischen Go und JavaScript, die eine nahtlose Kommunikation ohne HTTP-Overhead, Prozessgrenzen oder Serialisierungsengpässe ermöglicht.

Diagram

Wichtige Erkenntnis: Kein HTTP, kein IPC, keine Prozessgrenzen. Nur direkte Funktionsaufrufe mit Typsicherheit.

Wenn Ihre Anwendung startet, scannt Wails Ihre Dienste:

type GreetService struct {
prefix string
}
func (g *GreetService) Greet(name string) string {
return g.prefix + name + "!"
}
func (g *GreetService) Add(a, b int) int {
return a + b
}
// Register service
app := application.New(application.Options{
Services: []application.Service{
application.NewService(&GreetService{prefix: "Hello, "}),
},
})

Was Wails tut:

  1. Scannt die Struktur nach exportierten Methoden
  2. Extrahiert Typinformationen (Parameter, Rückgabetypen)
  3. Erstellt ein Verzeichnis, das Methodennamen Funktionen zuordnet
  4. Generiert TypeScript-Bindings mit vollständigen Typdefinitionen

Wails generiert automatisch TypeScript-Bindings:

frontend/bindings/GreetService.ts
export function Greet(name: string): Promise<string>
export function Add(a: number, b: number): Promise<number>

Typzuordnung:

Go-TypTypeScript-Typ
stringstring
int, int32, int64number
float32, float64number
boolboolean
[]TT[]
map[string]TRecord<string, T>
structinterface
time.TimeDate
errorException (geworfen)

Der Entwickler ruft die Go-Methode aus JavaScript heraus auf:

import { Greet, Add } from './bindings/GreetService'
// Call Go from JavaScript
const greeting = await Greet("World")
console.log(greeting) // "Hello, World!"
const sum = await Add(5, 3)
console.log(sum) // 8

Was passiert:

  1. Binding-Funktion aufgerufen - Greet("World")
  2. Nachricht erstellt - { service: "GreetService", method: "Greet", args: ["World"] }
  3. An die Brücke gesendet - Über die JavaScript-Brücke von WebView
  4. Promise zurückgegeben - Wartet auf Antwort

Die Brücke empfängt die Nachricht und verarbeitet sie:

Diagram

Sicherheit: Nur registrierte Dienste und exportierte Methoden sind aufrufbar.

Die Go-Methode wird ausgeführt:

func (g *GreetService) Greet(name string) string {
// This runs in Go
return g.prefix + name + "!"
}

Ausführungskontext:

  • Läuft in einer Goroutine (nicht-blockierend)
  • Hat Zugriff auf alle Go-Funktionen (Dateisystem, Netzwerk, Datenbanken)
  • Kann frei anderen Go-Code aufrufen
  • Gibt Ergebnis oder Fehler zurück

Das Ergebnis wird an JavaScript zurückgesendet:

// Promise resolves with result
const greeting = await Greet("World")
// greeting = "Hello, World!"

Fehlerbehandlung:

func (g *GreetService) Divide(a, b float64) (float64, error) {
if b == 0 {
return 0, errors.New("division by zero")
}
return a / b, nil
}
try {
const result = await Divide(10, 0)
} catch (error) {
console.error("Go error:", error) // "division by zero"
}

Typische Aufruf-Overhead: <1ms

Frontend Call → Bridge → Go Execution → Bridge → Frontend Response
↓ ↓ ↓ ↓ ↓
&lt;0.1ms &lt;0.1ms [varies] &lt;0.1ms &lt;0.1ms

Im Vergleich zu Alternativen:

  • HTTP/REST: 5-50ms (Netzwerkstack, Serialisierung)
  • IPC: 1-10ms (Prozessgrenzen, Marshalling)
  • Wails Bridge: <1ms (im Speicher, direkter Aufruf)

Overhead pro Aufruf: ~1KB (Nachrichtenpuffer)

Zero-Copy-Optimierung: Große Daten (>1MB) verwenden bei Gelegenheit shared Memory.

Aufrufe sind nebenläufig:

  • Jeder Aufruf läuft in seiner eigenen Goroutine
  • Mehrere Aufrufe können gleichzeitig ausgeführt werden
  • Keine Blockierung zwischen Aufrufen
// These run concurrently
const [result1, result2, result3] = await Promise.all([
SlowOperation1(),
SlowOperation2(),
SlowOperation3(),
])
// Go
func Example(
s string,
i int,
f float64,
b bool,
) (string, int, float64, bool) {
return s, i, f, b
}
// TypeScript (auto-generated)
function Example(
s: string,
i: number,
f: number,
b: boolean,
): Promise<[string, number, number, boolean]>
// Go
func Sum(numbers []int) int {
total := 0
for _, n := range numbers {
total += n
}
return total
}
// TypeScript
function Sum(numbers: number[]): Promise<number>
// Usage
const total = await Sum([1, 2, 3, 4, 5]) // 15
// Go
func GetConfig() map[string]interface{} {
return map[string]interface{}{
"theme": "dark",
"fontSize": 14,
"enabled": true,
}
}
// TypeScript
function GetConfig(): Promise<Record<string, any>>
// Usage
const config = await GetConfig()
console.log(config.theme) // "dark"
// Go
type User struct {
ID int `json:"id"`
Name string `json:"name"`
Email string `json:"email"`
}
func GetUser(id int) (*User, error) {
return &User{
ID: id,
Name: "Alice",---
**Fragen zur Brücke?** Fragen Sie im [Discord](https://discord.gg/JDdSxwjhGf) oder überprüfen Sie die [Bindungsbeispiele](https://github.com/wailsapp/wails/tree/master/v3/examples/binding).