Dieser Inhalt wurde automatisch aus dem Englischen übersetzt, und kann Fehler enthalten. Erfahre mehr über dieses Experiment.

View in English Always switch to English

set: Wasm-Tabelle-Anweisung

Baseline Widely available

This feature is well established and works across many devices and browser versions. It’s been available across browsers since November 2021.

Die table.set Tabelle-Anweisung ändert den gespeicherten Wert in einem bestimmten Tabellenelement.

Probieren Sie es aus

(module
  ;; Define function type
  (type $ret_i32 (func (result i32)))

  ;; table with 0 function slots
  (table $return_funcs 0 funcref)

  ;; Define a function of the specified type
  (func $f1 (type $ret_i32)
    (i32.const 42)
  )

  (elem declare func $f1)

  (func $populate
    (table.grow $return_funcs
      ref.null func
      (i32.const 1)
    )
    (drop)
    (table.set $return_funcs
      (i32.const 0)
      (ref.func $f1)
    )
  )

  (func (export "run") (result i32)
    (call $populate)
    (call_indirect (type $ret_i32) (i32.const 0))
  )
)
WebAssembly.instantiateStreaming(fetch("{%wasm-url%}")).then((result) => {
  const value = result.instance.exports.run();
  console.log(value);
});

Syntax

table.set identifier
table.set

Der table.set Anweisungstyp. Muss immer zuerst eingefügt werden.

identifier Optional

Ein Bezeichner für die Tabelle, in der Sie eine Referenz speichern möchten. Dies kann Folgendes sein:

name

Ein identifizierender Name für die Tabelle festgelegt, als sie erstmals erstellt wurde. Dieser muss mit einem $-Symbol beginnen, zum Beispiel $my_table.

index

Die Indexnummer der Tabelle, zum Beispiel 0 für die erste Tabelle im Wasm-Skript, 1 für die zweite, etc.

Wenn der identifier weggelassen wird, wird er standardmäßig auf 0 gesetzt.

Typ

[index, value] -> []
index

Der Elementindex, in dem der Wert gespeichert werden soll. Dies muss ein i32 Wert sein, zum Beispiel (i32.const 1).

value

Der Wert, der in der Tabelle gespeichert werden soll. Dies muss dem Elementtyp entsprechen, mit dem die Tabelle definiert wurde.

Traps

table.set löst einen Fehler aus, wenn:

Opcodes

Anweisung Binärer Opcode
table.set 𝟶𝚡𝟸𝟼 (variable-width LEB128)

Beschreibung

Die table.set Anweisung wird verwendet, um eine bestimmte Funktion in einem bestimmten Element einer bestehenden Tabelle zu speichern. Insbesondere ermöglicht es Wasm-Modulen, Referenzen, die in Tabellen enthalten sind, zur Laufzeit zu verändern.

Dies ist praktisch, wenn im späteren Verlauf des Programms unterschiedliche Referenzen erforderlich sind, als diejenigen, mit denen die Tabelle initialisiert wurde. Zum Beispiel könnten Sie eine Tabelle mit einem Element erstellen, zwei Funktionen definieren und dann das Tabellenelement mit einer Referenz auf eine dieser Funktionen mithilfe der elem Moduldefinition initialisieren:

wat
(module
  ;; Create table with 1 function slot
  (table $return_funcs 1 funcref)

  ;; Define basic functions that return i32s
  (func $f1 (result i32)
    (i32.const 42)
  )
  (func $f2 (result i32)
    (i32.const 100)
  )

  ;; initialize table slot
  (elem (i32.const 0) $f1)

  ...

Beim Verwenden von Anweisungen wie table.set, müssen auch die Funktionen im Voraus deklariert werden, indem (elem declare ...) verwendet wird, damit sie später referenziert werden können:

wat
  ...

  (elem declare func $f1 $f2)

  ...

Später können Sie dann die in dem Tabellenelement gespeicherte Funktionsreferenz dynamisch mithilfe von table.set ändern:

wat
  ...

  (func $populate
    (table.set $return_funcs
      (i32.const 0)
      (ref.func $f2)
    )
  )

  ...

Eine Wasm-Tabelle kann von JavaScript aus mit der Methode table.set() vergrößert werden.

Beispiele

Erstellen, Vergrößern und Setzen einer Tabelle

Dieses Beispiel zeigt, wie man eine Tabelle erstellt, ihre Größe vergrößert, die in ihr gespeicherten Funktionen dynamisch ändert und dann die Funktion aufruft, die zu jedem Zeitpunkt in der Tabelle gespeichert ist.

JavaScript

In unserem Skript beginnen wir, indem wir eine Referenz auf ein <p>-Element greifen, in das wir Ergebnisse ausgeben werden. Dann definieren wir ein obj-Objekt mit einer Funktion namens output(), die einen gegebenen Wert dem textContent eines gegebenen Elements hinzufügt.

Wir kompilieren und instanziieren dann unser Wasm-Modul mit der Methode WebAssembly.instantiateStreaming(), wobei wir das obj-Objekt importieren.

Wenn das Ergebnis zurückgegeben wird, rufen wir die exportierte Wasm run()-Funktion auf, die im WebAssembly Instance exports Objekt verfügbar ist, und übergeben ihr das outputElem-Element als Parameter.

js
const outputElem = document.querySelector("p");

const obj = {
  output: function (elem, val) {
    elem.textContent += `${val} `;
  },
};

WebAssembly.instantiateStreaming(fetch("{%wasm-url%}"), {
  obj,
}).then((result) => {
  value = result.instance.exports.run(outputElem);
});

Wasm

In unserem Wasm-Modul importieren wir zuerst die JavaScript output()-Funktion, wobei wir sicherstellen, dass sie zwei Parameter hat, ein externref und ein i32.

Als nächstes definieren wir einen Funktionstyp namens $ret_i32, der einen i32-Wert zurückgibt. Dann definieren wir zwei Funktionen basierend auf diesem Typ namens $f1 und $f2, die die innerhalb definierten Werte zurückgeben, und deklarieren sie im Voraus mit (elem declare func $f1 $f2), damit sie später referenziert werden können. Danach definieren wir eine table namens $func_table, die Funktionsreferenzen speichert (daher wird funcref angegeben) und anfänglich leer ist.

Schließlich exportieren wir die run()-Funktion, die ein externref namens $elem als Parameter nimmt. Im Funktionskörper:

  • Nutzen wir table.grow, um die Tabellengröße um 1 wachsen zu lassen, mit einem initialen ref.null Wert, und überprüfen, ob das Operationsergebnis -1 ist, was auf ein Scheitern hinweisen würde.
  • Setzen unser Tabellenelement, um die $f1-Funktion mit table.set zu enthalten, und rufen die importierte $output-Funktion auf, indem wir ihr als Parameter das $elem externref, das in die output()-Funktion übergeben wurde, und den Wert, der von der $f1-Funktion zurückgegeben wird, der aus der Tabelle mit (call_indirect (type $ret_i32) (i32.const 0)) referenziert wird, übergeben.
  • Setzen unser Tabellenelement, um die $f2-Funktion mit table.set zu enthalten, und rufen die output()-Funktion erneut auf.
wat
(module
  ;; Import output function
  (import "obj" "output" (func $output (param externref) (param i32)))

  ;; Define function type
  (type $ret_i32 (func (result i32)))

  ;; Define basic functions that return i32s
  (func $f1 (result i32)
    (i32.const 42)
  )
  (func $f2 (result i32)
    (i32.const 100)
  )

  (elem declare func $f1 $f2)

  ;; Define an initially empty table of funcrefs
  (table $func_table 0 funcref)

  (func (export "run") (param $elem externref)
    ;; Grow the table by 1, setting the initial value to null.
    ;; Check the result for -1, which indicates failure.
    (if
      (i32.eq
        (table.grow $func_table
          ref.null func
          (i32.const 1)
        )
        i32.const -1
      )

      ;; Trap if we failed to grow the table
      (then unreachable)
    )

    ;; Set the first function in the table to f1
    (table.set $func_table
        (i32.const 0)
        (ref.func $f1)
    )

    ;; Call the output function, to output the table
    ;; function's return value to the DOM
    (call $output
      (local.get $elem)
      (call_indirect (type $ret_i32) (i32.const 0))
    )

    ;; Set the first function in the table to f2
    (table.set $func_table
        (i32.const 0)
        (ref.func $f2)
    )

    ;; Call the output function, to output the table
    ;; function's return value to the DOM
    (call $output
      (local.get $elem)
      (call_indirect (type $ret_i32) (i32.const 0))
    )
  )
)

Ergebnis

Die Ausgabe ist wie folgt:

Das ist sinnvoll, da jedes Mal, wenn die output()-Funktion aus dem Wasm-Modul ausgeführt wird, der Wert, der als zweiter Parameter in sie übergeben wird, in unser Ergebnis-<p> im DOM gedruckt wird. Jeder Wert ist der zurückgegebene Wert der $f1 und $f2 Funktionen — 42 und 100 jeweils.

Spezifikationen

Specification
Unknown specification
# syntax-instr-table

Browser-Kompatibilität

Siehe auch