Template Literals (Template Strings)
Template Literals sind Literale, die durch Backticks (`
) abgegrenzt werden. Sie ermöglichen mehrzeilige Zeichenfolgen, Zeichenfolgeninterpolation mit eingebetteten Ausdrücken und spezielle Konstrukte namens getaggte Templates.
Template Literals werden manchmal informell Template Strings genannt, da sie am häufigsten für die Zeichenfolgeninterpolation verwendet werden (um Zeichenfolgen durch Ersetzen von Platzhaltern zu erstellen). Ein getaggtes Template Literal muss jedoch nicht in einer Zeichenfolge resultieren; es kann mit einer benutzerdefinierten Tag-Funktion verwendet werden, um beliebige Operationen auf den verschiedenen Teilen des Template Literals auszuführen.
Syntax
`string text`
`string text line 1
string text line 2`
`string text ${expression} string text`
tagFunction`string text ${expression} string text`
Parameter
string text
-
Der Zeichenfolgen-Text, der Teil des Template Literals wird. Fast alle Zeichen sind wörtlich erlaubt, einschließlich Zeilenumbrüche und anderer Leerzeichen. Ungültige Escape-Sequenzen verursachen jedoch einen Syntaxfehler, es sei denn, es wird eine Tag-Funktion verwendet.
expression
-
Ein in die aktuelle Position einzufügender Ausdruck, dessen Wert in eine Zeichenfolge konvertiert oder an
tagFunction
übergeben wird. tagFunction
-
Falls angegeben, wird sie mit dem Template-Strings-Array und den Substitutionsexpressionen aufgerufen, und der Rückgabewert wird zum Wert des Template Literals. Siehe getaggte Templates.
Beschreibung
Template Literals werden von Backticks (`
) anstelle von doppelten oder einfachen Anführungszeichen eingeschlossen.
Zusätzlich zu normalen Zeichenfolgen können Template Literals auch andere Teile namens Platzhalter enthalten, die eingebettete Ausdrücke sind, die durch ein Dollarzeichen und geschweifte Klammern abgegrenzt sind: ${expression}
. Die Zeichenfolgen und Platzhalter werden an eine Funktion übergeben – entweder eine Standardfunktion oder eine von Ihnen bereitgestellte. Die Standardfunktion (wenn Sie keine eigene angeben) führt einfach eine Zeichenfolgeninterpolation durch, um die Platzhalter zu ersetzen, und verkettet dann die Teile zu einer einzigen Zeichenfolge.
Um eine eigene Funktion bereitzustellen, stellen Sie dem Template Literal einen Funktionsnamen voran; das Ergebnis wird als getaggtes Template bezeichnet. In diesem Fall wird das Template Literal an Ihre Tag-Funktion übergeben, in der Sie dann beliebige Operationen auf den verschiedenen Teilen des Template Literals ausführen können.
Um einen Backtick in einem Template Literal zu escapen, setzen Sie einen Backslash (\
) vor den Backtick.
`\`` === "`"; // true
Dollarzeichen können ebenfalls escaped werden, um die Interpolation zu verhindern.
`\${1}` === "${1}"; // true
Mehrzeilige Zeichenfolgen
Alle im Quelltext eingefügten neuen Zeilenzeichen sind Teil des Template Literals.
Bei Verwendung normaler Zeichenfolgen müssten Sie die folgende Syntax verwenden, um mehrzeilige Zeichenfolgen zu erhalten:
console.log("string text line 1\n" + "string text line 2");
// "string text line 1
// string text line 2"
Mit Template Literals können Sie dasselbe mit Folgendem erreichen:
console.log(`string text line 1
string text line 2`);
// "string text line 1
// string text line 2"
Wie bei normalen Zeichenfolgen-Literalen können Sie eine einzeilige Zeichenfolge über mehrere Zeilen zur besseren Lesbarkeit des Quellcodes schreiben, indem Sie den Zeilenumbruch mit einem Backslash (\
) escapen:
console.log(`string text line 1 \
string text line 2`);
// "string text line 1 string text line 2"
Zeichenfolgeninterpolation
Ohne Template Literals, wenn Sie Ausgaben von Ausdrücken mit Zeichenfolgen kombinieren möchten, würden Sie sie konkatenieren mit dem Additionsoperator +
:
const a = 5;
const b = 10;
console.log("Fifteen is " + (a + b) + " and\nnot " + (2 * a + b) + ".");
// "Fifteen is 15 and
// not 20."
Das kann schwer lesbar sein – besonders wenn Sie mehrere Ausdrücke haben.
Mit Template Literals können Sie den Konkatenationsoperator vermeiden – und die Lesbarkeit Ihres Codes verbessern – indem Sie Platzhalter in der Form ${expression}
verwenden, um Ersetzungen für eingebettete Ausdrücke vorzunehmen:
const a = 5;
const b = 10;
console.log(`Fifteen is ${a + b} and
not ${2 * a + b}.`);
// "Fifteen is 15 and
// not 20."
Beachten Sie, dass es einen leichten Unterschied zwischen den beiden Syntaxen gibt. Template Literals zwingen ihre Ausdrücke direkt in Zeichenfolgen, während die Addition ihre Operanden zuerst zu Primitiven zwingt. Für weitere Informationen siehe die Referenzseite für den +
Operator.
Verschachtelung von Templates
In bestimmten Fällen ist das Verschachteln eines Templates der einfachste (und möglicherweise lesbarere) Weg, um konfigurierbare Zeichenfolgen zu haben. Innerhalb eines durch Backticks abgegrenzten Templates ist es einfach, innere Backticks zuzulassen, indem Sie sie innerhalb eines ${expression}
Platzhalters im Template verwenden.
Ohne Template Literals könnten Sie zum Beispiel, wenn Sie einen bestimmten Wert basierend auf einer bestimmten Bedingung zurückgeben wollten, etwas wie das folgende tun:
let classes = "header";
classes += isLargeScreen()
? ""
: item.isCollapsed
? " icon-expander"
: " icon-collapser";
Mit einem Template Literal aber ohne Verschachtelung könnten Sie dies tun:
const classes = `header ${
isLargeScreen() ? "" : item.isCollapsed ? "icon-expander" : "icon-collapser"
}`;
Mit Verschachtelung von Template Literals können Sie dies tun:
const classes = `header ${
isLargeScreen() ? "" : `icon-${item.isCollapsed ? "expander" : "collapser"}`
}`;
Getaggte Templates
Eine fortgeschrittenere Form von Template Literals sind getaggte Templates.
Tags erlauben Ihnen, Template Literals mit einer Funktion zu analysieren. Das erste Argument einer Tag-Funktion enthält ein Array von Zeichenfolgenwerten. Die restlichen Argumente beziehen sich auf die Ausdrücke.
Die Tag-Funktion kann dann beliebige Operationen auf diesen Argumenten ausführen und die manipulierte Zeichenfolge zurückgeben. (Alternativ kann sie etwas völlig anderes zurückgeben, wie in einem der folgenden Beispiele beschrieben.)
Der Name der Funktion, die für das Tag verwendet wird, kann beliebig sein.
const person = "Mike";
const age = 28;
function myTag(strings, personExp, ageExp) {
const str0 = strings[0]; // "That "
const str1 = strings[1]; // " is a "
const str2 = strings[2]; // "."
const ageStr = ageExp < 100 ? "youngster" : "centenarian";
// We can even return a string built using a template literal
return `${str0}${personExp}${str1}${ageStr}${str2}`;
}
const output = myTag`That ${person} is a ${age}.`;
console.log(output);
// That Mike is a youngster.
Das Tag muss kein einfacher Bezeichner sein. Sie können jeden Ausdruck mit einem Vorrang von mehr als 16 verwenden, einschließlich Eigenschaftszugriff, Funktionsaufruf, Neuer Ausdruck, oder sogar ein weiteres getaggtes Template Literal.
console.log`Hello`; // [ 'Hello' ]
console.log.bind(1, 2)`Hello`; // 2 [ 'Hello' ]
new Function("console.log(arguments)")`Hello`; // [Arguments] { '0': [ 'Hello' ] }
function recursive(strings, ...values) {
console.log(strings, values);
return recursive;
}
recursive`Hello``World`;
// [ 'Hello' ] []
// [ 'World' ] []
Obwohl technisch durch die Syntax erlaubt, sind ungetaggte Template Literals Zeichenfolgen und werden einen TypeError
werfen, wenn sie verkettet sind.
console.log(`Hello``World`); // TypeError: "Hello" is not a function
Die einzige Ausnahme ist das optionale Verkettung, das einen Syntaxfehler werfen wird.
console.log?.`Hello`; // SyntaxError: Invalid tagged template on optional chain
console?.log`Hello`; // SyntaxError: Invalid tagged template on optional chain
Beachten Sie, dass diese beiden Ausdrücke immer noch analysierbar sind. Das bedeutet, dass sie nicht der automatischen Semikolon-Einfügung unterliegen, die nur Semikolons einfügt, um nicht-parsierbaren Code zu korrigieren.
// Still a syntax error
const a = console?.log
`Hello`
Tag-Funktionen müssen nicht einmal eine Zeichenfolge zurückgeben!
function template(strings, ...keys) {
return (...values) => {
const dict = values[values.length - 1] || {};
const result = [strings[0]];
keys.forEach((key, i) => {
const value = Number.isInteger(key) ? values[key] : dict[key];
result.push(value, strings[i + 1]);
});
return result.join("");
};
}
const t1Closure = template`${0}${1}${0}!`;
// const t1Closure = template(["","","","!"],0,1,0);
t1Closure("Y", "A"); // "YAY!"
const t2Closure = template`${0} ${"foo"}!`;
// const t2Closure = template([""," ","!"],0,"foo");
t2Closure("Hello", { foo: "World" }); // "Hello World!"
const t3Closure = template`I'm ${"name"}. I'm almost ${"age"} years old.`;
// const t3Closure = template(["I'm ", ". I'm almost ", " years old."], "name", "age");
t3Closure("foo", { name: "MDN", age: 30 }); // "I'm MDN. I'm almost 30 years old."
t3Closure({ name: "MDN", age: 30 }); // "I'm MDN. I'm almost 30 years old."
Das erste Argument, das von der Tag-Funktion empfangen wird, ist ein Array von Zeichenfolgen. Für jedes Template Literal entspricht seine Länge der Anzahl der Substitutionen (Vorkommen von ${…}
) plus eins und ist daher immer nicht leer.
Für jeden bestimmten getaggten Template Literal-Ausdruck wird die Tag-Funktion immer mit demselben Literale-Array aufgerufen, egal wie oft das Literal ausgewertet wird.
const callHistory = [];
function tag(strings, ...values) {
callHistory.push(strings);
// Return a freshly made object
return {};
}
function evaluateLiteral() {
return tag`Hello, ${"world"}!`;
}
console.log(evaluateLiteral() === evaluateLiteral()); // false; each time `tag` is called, it returns a new object
console.log(callHistory[0] === callHistory[1]); // true; all evaluations of the same tagged literal would pass in the same strings array
Dies erlaubt dem Tag, das Ergebnis basierend auf der Identität seines ersten Arguments zwischenzuspeichern. Um die Stabilität des Array-Werts weiter zu gewährleisten, sind sowohl das erste Argument als auch seine raw
Eigenschaft eingefroren, sodass Sie sie in keiner Weise ändern können.
Rohe Zeichenfolgen
Die spezielle raw
Eigenschaft, die im ersten Argument der Tag-Funktion verfügbar ist, ermöglicht Ihnen den Zugriff auf die rohen Zeichenfolgen, wie sie eingegeben wurden, ohne Verarbeitung von Escape-Sequenzen.
function tag(strings) {
console.log(strings.raw[0]);
}
tag`string text line 1 \n string text line 2`;
// Logs "string text line 1 \n string text line 2",
// including the two characters '\' and 'n'
Zusätzlich existiert die Methode String.raw()
, um rohe Zeichenfolgen genauso zu erstellen, wie es die Standardtemplate-Funktion und Zeichenfolgenkonkatenation tun würden.
const str = String.raw`Hi\n${2 + 3}!`;
// "Hi\\n5!"
str.length;
// 6
Array.from(str).join(",");
// "H,i,\\,n,5,!"
String.raw
funktioniert wie ein "Identitäts"-Tag, wenn das Literal keine Escape-Sequenzen enthält. Falls Sie ein tatsächliches Identitäts-Tag möchten, das immer so funktioniert, als wäre das Literal ungetaggt, können Sie eine benutzerdefinierte Funktion erstellen, die das "verarbeitete" (d.h. Escape-Sequenzen werden verarbeitet) Literal-Array an String.raw
übergibt, wobei vorgegeben wird, es seien rohe Zeichenfolgen.
const identity = (strings, ...values) =>
String.raw({ raw: strings }, ...values);
console.log(identity`Hi\n${2 + 3}!`);
// Hi
// 5!
Dies ist nützlich für viele Tools, die Literalen mit einem bestimmten Namen eine besondere Behandlung geben.
const html = (strings, ...values) => String.raw({ raw: strings }, ...values);
// Some formatters will format this literal's content as HTML
const doc = html`<!doctype html>
<html lang="en-US">
<head>
<title>Hello</title>
</head>
<body>
<h1>Hello world!</h1>
</body>
</html>`;
Getaggte Templates und Escape-Sequenzen
In normalen Template Literals sind die Escape-Sequenzen in Zeichenfolgen-Literalen alle erlaubt. Jede andere schlecht geformte Escape-Sequenz ist ein Syntaxfehler. Dies schließt ein:
\
gefolgt von einer Dezimalziffer außer0
, oder\0
gefolgt von einer Dezimalziffer; zum Beispiel\9
und\07
(was eine veraltete Syntax ist)\x
gefolgt von weniger als zwei Hexadezimalziffern (einschließlich keiner); zum Beispiel\xz
\u
nicht gefolgt von{
und gefolgt von weniger als vier Hexadezimalziffern (einschließlich keiner); zum Beispiel\uz
\u{}
umschließend ein ungültiger Unicode-Codepunkt — es enthält eine nicht-hexadezimale Ziffer, oder sein Wert ist größer als10FFFF
; zum Beispiel\u{110000}
und\u{z}
Hinweis: \
gefolgt von anderen Zeichen, obwohl sie nutzlos sein können, da nichts escaped wird, sind keine Syntaxfehler.
Dies ist jedoch problematisch für getaggte Templates, die neben dem "verarbeiteten" Literal auch Zugriff auf die rohen Literale haben (Escape-Sequenzen werden wie eingegeben beibehalten).
Getaggte Templates ermöglichen das Einbetten von beliebigem Zeichenfolgeninhalt, wobei Escape-Sequenzen einem anderen Syntax unterliegen können. Betrachten Sie ein Beispiel, bei dem wir LaTeX-Quelltext in JavaScript über String.raw
einbetten. Wir möchten weiterhin in der Lage sein, LaTeX-Makros zu verwenden, die mit u
oder x
beginnen, ohne den Syntaxbeschränkungen von JavaScript zu folgen. Daher wird die Syntaxbeschränkung für wohlgeformte Escape-Sequenzen aus getaggten Templates entfernt. Das folgende Beispiel verwendet MathJax, um LaTeX in einem Element darzustellen:
const node = document.getElementById("formula");
MathJax.typesetClear([node]);
// Throws in older ECMAScript versions (ES2016 and earlier)
// SyntaxError: malformed Unicode character escape sequence
node.textContent = String.raw`$\underline{u}$`;
MathJax.typesetPromise([node]);
Illegale Escape-Sequenzen müssen jedoch weiterhin in der "verarbeiteten" Darstellung dargestellt werden. Sie werden als undefined
Element im "verarbeiteten" Array angezeigt:
function log(str) {
console.log("Cooked:", str[0]);
console.log("Raw:", str.raw[0]);
}
log`\unicode`;
// Cooked: undefined
// Raw: \unicode
Beachten Sie, dass die Einschränkung der Escape-Sequenz nur aus getaggten Templates entfernt wird, nicht aber aus ungetaggten Template Literalen:
const bad = `bad escape sequence: \unicode`;
Spezifikationen
Specification |
---|
ECMAScript Language Specification # sec-template-literals |
Browser-Kompatibilität
BCD tables only load in the browser
Siehe auch
- Textformatierungsleitfaden
String
String.raw()
- Lexikalische Grammatik
- ES6 in Depth: Template Strings auf hacks.mozilla.org (2015)