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

Baseline Weitgehend verfügbar *

Diese Funktion ist gut etabliert und funktioniert auf vielen Geräten und in vielen Browserversionen. Sie ist seit Juli 2015 browserübergreifend verfügbar.

* Einige Teile dieser Funktion werden möglicherweise unterschiedlich gut unterstützt.

Das Set-Objekt ermöglicht es Ihnen, eindeutige Werte beliebigen Typs zu speichern, egal ob primitive Werte oder Objektverweise.

Beschreibung

Set-Objekte sind Sammlungen von Werten. Ein Wert im Set darf nur einmal vorkommen; er ist einzigartig in der Sammlung des Sets. Sie können die Elemente eines Sets in der Einfügereihenfolge durchlaufen. Die Einfügereihenfolge entspricht der Reihenfolge, in der jedes Element mit der add() Methode erfolgreich in das Set eingefügt wurde (das heißt, es war kein identisches Element bereits im Set vorhanden, als add() aufgerufen wurde).

Die Spezifikation erfordert, dass Sets so implementiert werden, dass sie im Durchschnitt Zugriffzeiten bieten, die sublinear zur Anzahl der Elemente in der Sammlung sind. Daher könnte es intern als Hashtabelle (mit O(1)-Suche), Suchbaum (mit O(log(N))-Suche) oder jeder anderen Datenstruktur dargestellt sein, solange die Komplexität besser als O(N) ist.

Wertgleichheit

Die Wertgleichheit basiert auf dem SameValueZero Algorithmus. (Früher wurde SameValue verwendet, was 0 und -0 als verschieden behandelte. Prüfen Sie die Browser-Kompatibilität.) Das bedeutet, NaN wird als gleich NaN angesehen (obwohl NaN !== NaN) und alle anderen Werte gelten als gleich gemäß der Semantik des === Operators. Auch für Objektschlüssel basiert die Gleichheit auf der Objektidentität. Sie werden durch Referenz und nicht durch Wert verglichen. Siehe Verwendung des Set-Objekts für Beispiele.

Leistung

Die has-Methode prüft, ob ein Wert im Set ist, und nutzt dabei einen Ansatz, der im Durchschnitt schneller ist als das Testen der meisten der zuvor hinzugefügten Elemente im Set. Insbesondere ist sie im Durchschnitt schneller als die Array.prototype.includes-Methode, wenn ein Array die gleiche Länge wie die Größe eines Sets hat.

Set-Zusammenstellung

Das Set-Objekt bietet einige Methoden, die Ihnen erlauben, Sets wie mathematische Operationen zu kombinieren. Diese Methoden umfassen:

Methode Rückgabewert Mathematisches Äquivalent Venn-Diagramm
A.difference(B) Set ABA\setminus B Ein Venn-Diagramm, bei dem sich zwei Kreise überschneiden. Die Differenz von A und B ist der Teil von A, der sich nicht mit B überschneidet.
A.intersection(B) Set ABA\cap B Ein Venn-Diagramm, bei dem sich zwei Kreise überschneiden. Der Schnittpunkt von A und B ist der Bereich, in dem sie sich überschneiden.
A.symmetricDifference(B) Set (AB)(BA)(A\setminus B)\cup(B\setminus A) Ein Venn-Diagramm, bei dem sich zwei Kreise überschneiden. Die symmetrische Differenz von A und B ist der Bereich, der entweder vom einen oder vom anderen Kreis, aber nicht von beiden eingeschlossen wird.
A.union(B) Set ABA\cup B Ein Venn-Diagramm, bei dem sich zwei Kreise überschneiden. Die Vereinigung von A und B ist der Bereich, der entweder von beiden oder einem der beiden Kreise eingeschlossen wird.
A.isDisjointFrom(B) Boolean AB=A\cap B = \empty Ein Venn-Diagramm mit zwei Kreisen. A und B sind disjunkt, weil die Kreise keinen Überlappungsbereich haben.
A.isSubsetOf(B) Boolean ABA\subseteq B Ein Venn-Diagramm mit zwei Kreisen. A ist eine Teilmenge von B, weil A vollständig in B enthalten ist.
A.isSupersetOf(B) Boolean ABA\supseteq B Ein Venn-Diagramm mit zwei Kreisen. A ist eine Obermenge von B, weil B vollständig in A enthalten ist.

Um sie universeller verwendbar zu machen, akzeptieren diese Methoden nicht nur Set-Objekte, sondern alles, was set-ähnlich ist.

Set-ähnliche Objekte

Alle Set-Zusammensetzungs-Methoden erfordern, dass this eine echte Set-Instanz ist, aber ihre Argumente müssen nur set-ähnlich sein. Ein set-ähnliches Objekt ist ein Objekt, das Folgendes bereitstellt:

  • Eine size Eigenschaft, die eine Zahl enthält.
  • Eine has() Methode, die ein Element entgegennimmt und einen Boolean zurückgibt.
  • Eine keys() Methode, die einen Iterator der Elemente im Set zurückgibt.

Zum Beispiel sind Map Objekte set-ähnlich, weil sie auch size, has(), und keys() haben, so dass sie sich in Set-Methoden verhalten wie Sets von Schlüsseln:

js
const a = new Set([1, 2, 3]);
const b = new Map([
  [1, "one"],
  [2, "two"],
  [4, "four"],
]);
console.log(a.union(b)); // Set(4) {1, 2, 3, 4}

Hinweis: Das set-ähnliche Protokoll ruft die keys()-Methode statt [Symbol.iterator]() auf, um Elemente zu produzieren. Dies dient dazu, Maps als gültige set-ähnliche Objekte zu etablieren, da für Maps der Iterator Einträge produziert, aber die has()-Methode Schlüssel nimmt.

Arrays sind nicht set-ähnlich, weil sie keine has()-Methode oder die size-Eigenschaft haben und ihre keys()-Methode Indizes statt Elemente produziert. WeakSet Objekte sind ebenfalls nicht set-ähnlich, weil sie keine keys()-Methode haben.

Set-artige Browser-APIs

Browser-Set-artige Objekte (oder "setlike objects") sind Web API Schnittstellen, die sich in vielerlei Hinsicht wie ein Set verhalten.

Genau wie Set können Elemente in der gleichen Reihenfolge durchlaufen werden, in der sie zum Objekt hinzugefügt wurden. Set-ähnliche Objekte und Set haben auch Eigenschaften und Methoden, die denselben Namen und dasselbe Verhalten teilen. Allerdings erlauben sie, anders als Set, nur einen spezifischen vordefinierten Typ für jeden Eintrag.

Die erlaubten Typen sind in der IDL-Definition der Spezifikation festgelegt. Zum Beispiel ist GPUSupportedFeatures ein Set-artiges Objekt, das Zeichenfolgen als Schlüssel/Wert verwenden muss. Dies ist in der IDL-Definition der Spezifikation unten definiert:

webidl
interface GPUSupportedFeatures {
  readonly setlike<DOMString>;
};

Set-ähnliche Objekte sind entweder schreibgeschützt oder schreibbar (siehe das readonly-Schlüsselwort in der IDL oben).

Die Methoden und Eigenschaften haben dasselbe Verhalten wie die entsprechenden Entitäten in Set, mit Ausnahme der Beschränkung auf die Typen des Eintrags.

Nachfolgend sind Beispiele für schreibgeschützte Set-artige Browser-Objekte:

Nachfolgend sind Beispiele für schreibbare Set-artige Browser-Objekte:

Konstruktor

Set()

Erstellt ein neues Set-Objekt.

Statische Eigenschaften

Set[Symbol.species]

Die Konstruktorfunktion, die zum Erstellen abgeleiteter Objekte verwendet wird.

Instanz-Eigenschaften

Diese Eigenschaften sind auf Set.prototype definiert und werden von allen Set-Instanzen geteilt.

Set.prototype.constructor

Die Konstruktorfunktion, die das Instanzobjekt erstellt hat. Für Set-Instanzen ist der Anfangswert der Set-Konstruktor.

Set.prototype.size

Gibt die Anzahl der Werte im Set-Objekt zurück.

Set.prototype[Symbol.toStringTag]

Der Anfangswert der [Symbol.toStringTag]-Eigenschaft ist der String "Set". Diese Eigenschaft wird in Object.prototype.toString() verwendet.

Instanz-Methoden

Set.prototype.add()

Fügt den angegebenen Wert in das Set ein, falls er noch nicht vorhanden ist.

Set.prototype.clear()

Entfernt alle Elemente aus dem Set-Objekt.

Set.prototype.delete()

Entfernt den angegebenen Wert aus diesem Set, falls er im Set ist.

Set.prototype.difference()

Nimmt ein Set und gibt ein neues Set zurück, das Elemente in diesem Set, aber nicht im angegebenen Set enthält.

Set.prototype.entries()

Gibt ein neues Iteratorobjekt zurück, das ein Array von [value, value] für jedes Element im Set-Objekt in Einfügereihenfolge enthält. Dies ist ähnlich dem Map-Objekt, so dass der Schlüssel jedes Eintrags derselbe ist wie der Wert für ein Set.

Set.prototype.forEach()

Ruft callbackFn einmal für jeden im Set-Objekt vorhandenen Wert in Einfügereihenfolge auf. Wenn ein thisArg-Parameter angegeben wird, wird er als this-Wert für jede Aufruf von callbackFn verwendet.

Set.prototype.has()

Gibt einen Boolean an, der zeigt, ob der angegebene Wert in diesem Set vorhanden ist oder nicht.

Set.prototype.intersection()

Nimmt ein Set und gibt ein neues Set zurück, das Elemente sowohl in diesem Set als auch im angegebenen Set enthält.

Set.prototype.isDisjointFrom()

Nimmt ein Set und gibt einen Boolean zurück, der anzeigt, ob dieses Set keine Elemente mit dem gegebenen Set gemeinsam hat.

Set.prototype.isSubsetOf()

Nimmt ein Set und gibt einen Boolean zurück, der anzeigt, ob alle Elemente dieses Sets im angegebenen Set sind.

Set.prototype.isSupersetOf()

Nimmt ein Set und gibt einen Boolean zurück, der anzeigt, ob alle Elemente des angegebenen Sets in diesem Set sind.

Set.prototype.keys()

Ein Alias für Set.prototype.values().

Set.prototype.symmetricDifference()

Nimmt ein Set und gibt ein neues Set zurück, das Elemente enthält, die entweder in diesem Set oder im angegebenen Set, aber nicht in beiden vorhanden sind.

Set.prototype.union()

Nimmt ein Set und gibt ein neues Set zurück, das Elemente enthält, die in entweder oder beiden dieses Sets und dem gegebenen Set vorhanden sind.

Set.prototype.values()

Gibt ein neues Iteratorobjekt zurück, das die Werte für jedes Element im Set-Objekt in Einfügereihenfolge liefert.

Set.prototype[Symbol.iterator]()

Gibt ein neues Iteratorobjekt zurück, das die Werte für jedes Element im Set-Objekt in Einfügereihenfolge liefert.

Beispiele

Verwendung des Set-Objekts

js
const mySet1 = new Set();

mySet1.add(1); // Set(1) { 1 }
mySet1.add(5); // Set(2) { 1, 5 }
mySet1.add(5); // Set(2) { 1, 5 }
mySet1.add("some text"); // Set(3) { 1, 5, 'some text' }
const o = { a: 1, b: 2 };
mySet1.add(o);

mySet1.add({ a: 1, b: 2 }); // o is referencing a different object, so this is okay

mySet1.has(1); // true
mySet1.has(3); // false, since 3 has not been added to the set
mySet1.has(5); // true
mySet1.has(Math.sqrt(25)); // true
mySet1.has("Some Text".toLowerCase()); // true
mySet1.has(o); // true

mySet1.size; // 5

mySet1.delete(5); // removes 5 from the set
mySet1.has(5); // false, 5 has been removed

mySet1.size; // 4, since we just removed one value

mySet1.add(5); // Set(5) { 1, 'some text', {...}, {...}, 5 } - a previously deleted item will be added as a new item, it will not retain its original position before deletion

console.log(mySet1); // Set(5) { 1, "some text", {…}, {…}, 5 }

Iteration über Sets

Die Iteration über ein Set besucht Elemente in der Einfügereihenfolge.

js
for (const item of mySet1) {
  console.log(item);
}
// 1, "some text", { "a": 1, "b": 2 }, { "a": 1, "b": 2 }, 5

for (const item of mySet1.keys()) {
  console.log(item);
}
// 1, "some text", { "a": 1, "b": 2 }, { "a": 1, "b": 2 }, 5

for (const item of mySet1.values()) {
  console.log(item);
}
// 1, "some text", { "a": 1, "b": 2 }, { "a": 1, "b": 2 }, 5

// key and value are the same here
for (const [key, value] of mySet1.entries()) {
  console.log(key);
}
// 1, "some text", { "a": 1, "b": 2 }, { "a": 1, "b": 2 }, 5

// Convert Set object to an Array object, with Array.from
const myArr = Array.from(mySet1); // [1, "some text", {"a": 1, "b": 2}, {"a": 1, "b": 2}, 5]

// the following will also work if run in an HTML document
mySet1.add(document.body);
mySet1.has(document.querySelector("body")); // true

// converting between Set and Array
const mySet2 = new Set([1, 2, 3, 4]);
console.log(mySet2.size); // 4
console.log([...mySet2]); // [1, 2, 3, 4]

// intersect can be simulated via
const intersection = new Set([...mySet1].filter((x) => mySet2.has(x)));

// difference can be simulated via
const difference = new Set([...mySet1].filter((x) => !mySet2.has(x)));

// Iterate set entries with forEach()
mySet2.forEach((value) => {
  console.log(value);
});
// 1
// 2
// 3
// 4

Implementierung grundlegender Set-Operationen

js
function isSuperset(set, subset) {
  for (const elem of subset) {
    if (!set.has(elem)) {
      return false;
    }
  }
  return true;
}

function union(setA, setB) {
  const _union = new Set(setA);
  for (const elem of setB) {
    _union.add(elem);
  }
  return _union;
}

function intersection(setA, setB) {
  const _intersection = new Set();
  for (const elem of setB) {
    if (setA.has(elem)) {
      _intersection.add(elem);
    }
  }
  return _intersection;
}

function symmetricDifference(setA, setB) {
  const _difference = new Set(setA);
  for (const elem of setB) {
    if (_difference.has(elem)) {
      _difference.delete(elem);
    } else {
      _difference.add(elem);
    }
  }
  return _difference;
}

function difference(setA, setB) {
  const _difference = new Set(setA);
  for (const elem of setB) {
    _difference.delete(elem);
  }
  return _difference;
}

// Examples
const setA = new Set([1, 2, 3, 4]);
const setB = new Set([2, 3]);
const setC = new Set([3, 4, 5, 6]);

isSuperset(setA, setB); // returns true
union(setA, setC); // returns Set {1, 2, 3, 4, 5, 6}
intersection(setA, setC); // returns Set {3, 4}
symmetricDifference(setA, setC); // returns Set {1, 2, 5, 6}
difference(setA, setC); // returns Set {1, 2}

Beziehung zu Arrays

js
const myArray = ["value1", "value2", "value3"];

// Use the regular Set constructor to transform an Array into a Set
const mySet = new Set(myArray);

mySet.has("value1"); // returns true

// Use the spread syntax to transform a set into an Array.
console.log([...mySet]); // Will show you exactly the same Array as myArray

Entfernen von doppelten Elementen aus einem Array

js
// Use to remove duplicate elements from an array
const numbers = [2, 13, 4, 4, 2, 13, 13, 4, 4, 5, 5, 6, 6, 7, 5, 32, 13, 4, 5];

console.log([...new Set(numbers)]); // [2, 13, 4, 5, 6, 7, 32]

Beziehung zu Zeichenfolgen

js
// Case sensitive (set will contain "F" and "f")
new Set("Firefox"); // Set(7) [ "F", "i", "r", "e", "f", "o", "x" ]

// Duplicate omission ("f" occurs twice in the string but set will contain only one)
new Set("firefox"); // Set(6) [ "f", "i", "r", "e", "o", "x" ]

Verwenden eines Sets zur Sicherstellung der Einzigartigkeit einer Werteliste

js
const array = Array.from(document.querySelectorAll("[id]")).map((e) => e.id);

const set = new Set(array);
console.assert(set.size === array.length);

Spezifikationen

Spezifikation
ECMAScript® 2027 Language Specification
# sec-set-objects

Browser-Kompatibilität

Siehe auch