Übertragbare Objekte
Übertragbare Objekte sind Objekte, die Ressourcen besitzen, die von einem Kontext in einen anderen übertragen werden können, wobei sichergestellt wird, dass die Ressourcen immer nur in einem Kontext verfügbar sind. Nach einer Übertragung ist das ursprüngliche Objekt nicht mehr verwendbar; es verweist nicht mehr auf die übertragene Ressource und jeder Versuch, das Objekt zu lesen oder zu schreiben, löst eine Ausnahme aus.
Übertragbare Objekte werden häufig verwendet, um Ressourcen zu teilen, die nur sicher einem einzelnen JavaScript-Thread zugänglich gemacht werden können. Zum Beispiel ist ein ArrayBuffer
ein übertragbares Objekt, das einen Speicherblock besitzt. Wenn ein solcher Puffer zwischen Threads übertragen wird, wird die zugehörige Speicherressource vom ursprünglichen Puffer getrennt und an das Pufferobjekt angehängt, das im neuen Thread erstellt wurde. Das Pufferobjekt im ursprünglichen Thread ist nicht mehr verwendbar, da es keine Speicherressource mehr besitzt.
Übertragung kann auch verwendet werden, wenn tiefe Kopien von Objekten mit structuredClone()
erstellt werden. Nach der Klonoperation werden die übertragenen Ressourcen verschoben, anstatt zum geklonten Objekt kopiert zu werden.
Für sowohl postMessage()
als auch structuredClone()
müssen übertragene Ressourcen an das Datenobjekt angehängt werden, da sie ansonsten am empfangenden Ende nicht verfügbar wären, weil das übertragbare Array nur anzeigt, wie bestimmte Ressourcen gesendet werden sollen, aber sie nicht tatsächlich sendet (obwohl sie immer abgetrennt würden).
Der Mechanismus, der verwendet wird, um die Ressourcen eines Objekts zu übertragen, hängt vom Objekt ab. Zum Beispiel, wenn ein ArrayBuffer
zwischen Threads übertragen wird, wird die Speicherressource, auf die er verweist, buchstäblich zwischen den Kontexten in einem schnellen und effizienten Zero-Copy-Verfahren bewegt. Andere Objekte können durch Kopieren der zugehörigen Ressource und anschließendes Löschen aus dem alten Kontext übertragen werden.
Nicht alle Objekte sind übertragbar. Eine Liste übertragbarer Objekte wird nachfolgend bereitgestellt.
Übertragung von Objekten zwischen Threads
Der folgende Code demonstriert, wie die Übertragung funktioniert, wenn eine Nachricht von einem Hauptthread an einen Web Worker-Thread gesendet wird. Das Uint8Array
wird im Worker kopiert (dupliziert), während sein Puffer übertragen wird. Nach der Übertragung wird jeder Versuch, uInt8Array
vom Hauptthread zu lesen oder zu schreiben, eine Ausnahme auslösen, aber Sie können immer noch die byteLength
überprüfen, um zu bestätigen, dass diese jetzt null ist.
// Create an 8MB "file" and fill it. 8MB = 1024 * 1024 * 8 B
const uInt8Array = new Uint8Array(1024 * 1024 * 8).map((v, i) => i);
console.log(uInt8Array.byteLength); // 8388608
// Transfer the underlying buffer to a worker
worker.postMessage(uInt8Array, [uInt8Array.buffer]);
console.log(uInt8Array.byteLength); // 0
Hinweis: Typed Arrays wie Int32Array
und Uint8Array
sind serialisierbar, aber nicht übertragbar. Ihr zugrunde liegender Puffer ist jedoch ein ArrayBuffer
, der ein übertragbares Objekt ist. Wir könnten uInt8Array.buffer
im Datenparameter senden, aber nicht uInt8Array
im Übertragungsarray.
Übertragung während einer Klonoperation
Der folgende Code zeigt eine structuredClone()
-Operation, bei der der zugrunde liegende Puffer vom Originalobjekt in den Klon kopiert wird.
const original = new Uint8Array(1024);
const clone = structuredClone(original);
console.log(original.byteLength); // 1024
console.log(clone.byteLength); // 1024
original[0] = 1;
console.log(clone[0]); // 0
// Transferring the Uint8Array would throw an exception as it is not a transferable object
// const transferred = structuredClone(original, {transfer: [original]});
// We can transfer Uint8Array.buffer.
const transferred = structuredClone(original, { transfer: [original.buffer] });
console.log(transferred.byteLength); // 1024
console.log(transferred[0]); // 1
// After transferring Uint8Array.buffer cannot be used.
console.log(original.byteLength); // 0
Unterstützte Objekte
Schnittstellen, die übertragbar sind, sollten diese Information in ihrer Einführung enthalten.
Einige der Elemente, die von verschiedenen Spezifikationen als übertragbar angegeben werden, sind unten aufgeführt (diese Liste ist möglicherweise nicht vollständig!):
ArrayBuffer
MessagePort
ReadableStream
WritableStream
TransformStream
WebTransportReceiveStream
WebTransportSendStream
AudioData
ImageBitmap
VideoFrame
OffscreenCanvas
RTCDataChannel
MediaSourceHandle
MIDIAccess
Hinweis:
Übertragbare Objekte sind in Web IDL-Dateien mit dem Attribut [Transferable]
gekennzeichnet. Browser-Unterstützung kann in den jeweiligen Angaben zur Browser-Kompatibilität des Objekts durch das transferable
Untermerkmal angegeben sein (siehe RTCDataChannel
für ein Beispiel).