Heim Web-Frontend js-Tutorial Abschlüsse in der funktionalen Javascript-Programmierung verstehen_Javascript-Kenntnisse

Abschlüsse in der funktionalen Javascript-Programmierung verstehen_Javascript-Kenntnisse

May 16, 2016 pm 03:11 PM
javascript 闭包

Closure ist ein Konzept in der funktionalen Programmierung, das in den 1960er Jahren auftauchte. Die früheste Sprache zur Implementierung von Closing war Scheme, ein Dialekt von LISP. Seitdem wurden Schließungsfunktionen in großem Umfang von anderen Sprachen übernommen.
Die strikte Definition eines Abschlusses ist „eine Menge, die aus einer Funktion (Umgebung) und ihren eingeschlossenen freien Variablen besteht.“ Diese Definition ist für jeden etwas unklar, daher erklären wir sie zunächst anhand von Beispielen und eine weniger strenge Erklärung ist ein Abschluss. und gibt dann Beispiele für einige klassische Verwendungen von Verschlüssen.

Was ist Schließung?

Laienhaft ausgedrückt ist jede Funktion in JavaScript ein Abschluss, aber im Allgemeinen können verschachtelte Funktionen
besser widerspiegeln Um die Eigenschaften des Verschlusses zu zeigen, sehen Sie sich bitte das folgende Beispiel an:

var generateClosure = function() {
var count = 0;
var get = function() {
count ++;
return count;
};
return get;
};
var counter = generateClosure();
console.log(counter()); // 输出 1
console.log(counter()); // 输出 2
console.log(counter()); // 输出 3
Nach dem Login kopieren

In diesem Code gibt es eine lokale Variablenanzahl in der Funktion genericClosure() mit einem Anfangswert von 0. Es gibt auch eine Funktion namens get, die die count-Variable in ihrem übergeordneten Bereich, die Funktion „generateClosure()“, um 1 erhöht und den Wert von count zurückgibt. Der Rückgabewert von genericClosure() ist die Get-Funktion. Extern haben wir die Funktion „generateClosure()“ über die Variable „counter“ aufgerufen und ihren Rückgabewert erhalten, der die Funktion „get“ ist. Dann haben wir mehrmals „counter()“ aufgerufen und festgestellt, dass der zurückgegebene Wert jedes Mal um 1 erhöht wurde.
Schauen wir uns die Merkmale des obigen Beispiels an. Nach dem üblichen Verständnis der imperativen Programmierung ist count eine Variable innerhalb der Funktion „generateClosure“. die Zählvariable Der angewendete Speicherplatz wird freigegeben. Das Problem besteht darin, dass counter() nach Beendigung des Aufrufs von genericClosure() auf die „bereits freigegebene“ Zählvariable verweist und nicht nur kein Fehler auftritt, sondern count auch bei jedem Aufruf von counter() geändert und zurückgegeben wird. Was ist los?
Genau das ist das Merkmal der sogenannten Schließung. Wenn eine Funktion eine darin definierte Funktion zurückgibt, wird ein Abschluss generiert. Der Abschluss umfasst nicht nur die zurückgegebene Funktion, sondern auch die Umgebung, in der die Funktion definiert ist. Wenn im obigen Beispiel die interne Funktion get der Funktion genericClosure() von einer externen Variablen counter referenziert wird, sind die lokalen Variablen von counter und genericClosure() ein Abschluss. Wenn es nicht klar genug ist, kann das folgende Beispiel helfen
Du verstehst:

var generateClosure = function() {
var count = 0;
var get = function() {
count ++;
return count;
};
return get;
};
var counter1 = generateClosure();
var counter2 = generateClosure();
console.log(counter1()); // 输出 1
console.log(counter2()); // 输出 1
console.log(counter1()); // 输出 2
console.log(counter1()); // 输出 3
console.log(counter2()); // 输出 2
Nach dem Login kopieren

Das obige Beispiel erklärt, wie Abschlüsse generiert werden: counter1 und counter2 rufen jeweils die Funktion genericClosure() auf und generieren zwei Instanzen von Abschlüssen. Die Zählvariablen, auf die sie intern verweisen, gehören zu ihren jeweiligen Betriebsumgebungen. Wir können verstehen, dass bei der Rückgabe der Get-Funktion durch genericClosure () die internen Variablen der Funktion „GenerateClosure ()“, auf die get verweisen kann (dh die Zählvariable), auch privat zurückgegeben werden und eine Kopie im Speicher generiert wird then genericClosure( ) Die beiden Instanzen der zurückgegebenen Funktion, counter1 und counter2, sind unabhängig voneinander.

Der Zweck der Schließung

1. Verschachtelte Rückruffunktion
Abschlüsse haben zwei Hauptzwecke: Zum einen werden verschachtelte Rückruffunktionen implementiert, zum anderen werden die Details eines Objekts ausgeblendet. Schauen wir uns zunächst das folgende Codebeispiel an, um verschachtelte Rückruffunktionen zu verstehen. Der folgende Code verwendet MongoDB in Node.js, um eine einfache Funktion zum Hinzufügen von Benutzern zu implementieren:

exports.add_user = function(user_info, callback) {
var uid = parseInt(user_info['uid']);
mongodb.open(function(err, db) {
if (err) {callback(err); return;}
db.collection('users', function(err, collection) {
if (err) {callback(err); return;}
collection.ensureIndex("uid", function(err) {
if (err) {callback(err); return;}
collection.ensureIndex("username", function(err) {
if (err) {callback(err); return;}
collection.findOne({uid: uid}, function(err) {
if (err) {callback(err); return;}
if (doc) {
callback('occupied');
} else {
var user = {
uid: uid,
user: user_info,
};
collection.insert(user, function(err) {
callback(err);
});
}
});
});
});
});
});
};
Nach dem Login kopieren

Wenn Sie mit Node.js oder MongoDB nicht vertraut sind, spielt es keine Rolle. Sie müssen die Details nicht verstehen, sondern nur die allgemeine Logik verstehen. Dieser Code verwendet Verschachtelungsebenen von Abschlüssen, und jede Verschachtelungsebene ist eine Rückruffunktion. Die Rückruffunktion wird nicht sofort ausgeführt, sondern von der angeforderten Funktion zurückgerufen, nachdem die entsprechende Anfrage verarbeitet wurde. Wir können sehen, dass es in jeder Verschachtelungsebene einen Verweis auf den Rückruf gibt und die innerste Ebene auch die von der äußeren Ebene definierte UID-Variable verwendet. Aufgrund des Abschlussmechanismus werden die in ihrem Bereich angewendeten Variablen nicht freigegeben, selbst wenn die äußere Funktion ausgeführt wurde, da die innere Funktion möglicherweise weiterhin auf diese Variablen verweist und so den verschachtelten asynchronen Rückruf perfekt realisiert.

2. Private Mitglieder implementieren
Wir wissen, dass JavaScript-Objekte keine privaten Eigenschaften haben, was bedeutet, dass jede Eigenschaft des Objekts der Außenwelt ausgesetzt ist. Dies kann zu Sicherheitsrisiken führen, wenn beispielsweise der Benutzer des Objekts ein Attribut direkt ändert, wodurch die Konsistenz der internen Daten des Objekts zerstört wird usw. JavaScript verwendet die Konvention, um vor allen privaten Eigenschaften (z. B. _myPrivateProp) einen Unterstrich zu setzen, um anzuzeigen, dass diese Eigenschaft privat ist und externe Objekte sie nicht direkt lesen oder schreiben sollten. Aber dies ist nur eine informelle Vereinbarung, vorausgesetzt, der Benutzer des Objekts tut dies nicht. Gibt es einen strengeren Mechanismus? Die Antwort lautet: Ja, dies kann durch Schließung erreicht werden. Schauen wir uns noch einmal das vorherige Beispiel an:

var generateClosure = function() {
var count = 0;
var get = function() {
count ++;
return count;
};
return get;
};
var counter = generateClosure();
console.log(counter()); // 输出 1
console.log(counter()); // 输出 2
console.log(counter()); // 输出 3
Nach dem Login kopieren

Wir können sehen, dass nur der Aufruf von counter() auf die Zählvariable im Abschluss zugreifen und diese gemäß den Regeln um 1 erhöhen kann. Es ist unmöglich, die Zählvariable auf andere Weise zu finden. Inspiriert durch dieses einfache Beispiel können wir ein Objekt mit einem Abschluss kapseln und nur ein „Accessor“-Objekt zurückgeben, um die Details zu verbergen.

Das Obige ist der gesamte Inhalt dieses Artikels. Ich hoffe, er kann jedem helfen, JavaScript-Schließungen besser zu lernen und zu verstehen.

Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn

Heiße KI -Werkzeuge

Undresser.AI Undress

Undresser.AI Undress

KI-gestützte App zum Erstellen realistischer Aktfotos

AI Clothes Remover

AI Clothes Remover

Online-KI-Tool zum Entfernen von Kleidung aus Fotos.

Undress AI Tool

Undress AI Tool

Ausziehbilder kostenlos

Clothoff.io

Clothoff.io

KI-Kleiderentferner

Video Face Swap

Video Face Swap

Tauschen Sie Gesichter in jedem Video mühelos mit unserem völlig kostenlosen KI-Gesichtstausch-Tool aus!

Heiße Werkzeuge

Notepad++7.3.1

Notepad++7.3.1

Einfach zu bedienender und kostenloser Code-Editor

SublimeText3 chinesische Version

SublimeText3 chinesische Version

Chinesische Version, sehr einfach zu bedienen

Senden Sie Studio 13.0.1

Senden Sie Studio 13.0.1

Leistungsstarke integrierte PHP-Entwicklungsumgebung

Dreamweaver CS6

Dreamweaver CS6

Visuelle Webentwicklungstools

SublimeText3 Mac-Version

SublimeText3 Mac-Version

Codebearbeitungssoftware auf Gottesniveau (SublimeText3)

Was bedeutet der Abschluss im C++-Lambda-Ausdruck? Was bedeutet der Abschluss im C++-Lambda-Ausdruck? Apr 17, 2024 pm 06:15 PM

In C++ ist ein Abschluss ein Lambda-Ausdruck, der auf externe Variablen zugreifen kann. Um einen Abschluss zu erstellen, erfassen Sie die äußere Variable im Lambda-Ausdruck. Abschlüsse bieten Vorteile wie Wiederverwendbarkeit, Ausblenden von Informationen und verzögerte Auswertung. Sie sind in realen Situationen nützlich, beispielsweise bei Ereignishandlern, bei denen der Abschluss auch dann noch auf die äußeren Variablen zugreifen kann, wenn diese zerstört werden.

Wie implementiert man einen Abschluss im C++-Lambda-Ausdruck? Wie implementiert man einen Abschluss im C++-Lambda-Ausdruck? Jun 01, 2024 pm 05:50 PM

C++-Lambda-Ausdrücke unterstützen Abschlüsse, die Funktionsbereichsvariablen speichern und sie für Funktionen zugänglich machen. Die Syntax lautet [capture-list](parameters)->return-type{function-body}. Capture-Liste definiert die zu erfassenden Variablen. Sie können [=] verwenden, um alle lokalen Variablen nach Wert zu erfassen, [&], um alle lokalen Variablen nach Referenz zu erfassen, oder [Variable1, Variable2,...], um bestimmte Variablen zu erfassen. Lambda-Ausdrücke können nur auf erfasste Variablen zugreifen, den ursprünglichen Wert jedoch nicht ändern.

Was sind die Vor- und Nachteile von Abschlüssen in C++-Funktionen? Was sind die Vor- und Nachteile von Abschlüssen in C++-Funktionen? Apr 25, 2024 pm 01:33 PM

Ein Abschluss ist eine verschachtelte Funktion, die auf Variablen im Bereich der äußeren Funktion zugreifen kann. Zu ihren Vorteilen gehören Datenkapselung, Zustandserhaltung und Flexibilität. Zu den Nachteilen gehören der Speicherverbrauch, die Auswirkungen auf die Leistung und die Komplexität des Debuggens. Darüber hinaus können Abschlüsse anonyme Funktionen erstellen und diese als Rückrufe oder Argumente an andere Funktionen übergeben.

Lösen Sie das durch Schließungen verursachte Speicherverlustproblem Lösen Sie das durch Schließungen verursachte Speicherverlustproblem Feb 18, 2024 pm 03:20 PM

Titel: Durch Abschlüsse und Lösungen verursachte Speicherlecks Einführung: Abschlüsse sind ein sehr verbreitetes Konzept in JavaScript, das internen Funktionen den Zugriff auf Variablen externer Funktionen ermöglicht. Allerdings können Schließungen bei falscher Verwendung zu Speicherverlusten führen. In diesem Artikel wird das durch Schließungen verursachte Speicherverlustproblem untersucht und Lösungen sowie spezifische Codebeispiele bereitgestellt. 1. Durch Schließungen verursachte Speicherlecks Das Merkmal von Schließungen besteht darin, dass interne Funktionen auf Variablen externer Funktionen zugreifen können, was bedeutet, dass in Schließungen referenzierte Variablen nicht durch Müll gesammelt werden. Bei unsachgemäßer Verwendung

Der Einfluss von Funktionszeigern und -abschlüssen auf die Golang-Leistung Der Einfluss von Funktionszeigern und -abschlüssen auf die Golang-Leistung Apr 15, 2024 am 10:36 AM

Die Auswirkungen von Funktionszeigern und -abschlüssen auf die Go-Leistung sind wie folgt: Funktionszeiger: Etwas langsamer als direkte Aufrufe, aber verbessert die Lesbarkeit und Wiederverwendbarkeit. Schließungen: Normalerweise langsamer, kapseln aber Daten und Verhalten. Praktischer Fall: Funktionszeiger können Sortieralgorithmen optimieren und Abschlüsse können Ereignishandler erstellen, aber sie bringen Leistungseinbußen mit sich.

Die Rolle des Golang-Funktionsschlusses beim Testen Die Rolle des Golang-Funktionsschlusses beim Testen Apr 24, 2024 am 08:54 AM

Funktionsabschlüsse der Go-Sprache spielen beim Unit-Testen eine wichtige Rolle: Werte erfassen: Abschlüsse können auf Variablen im äußeren Bereich zugreifen, sodass Testparameter erfasst und in verschachtelten Funktionen wiederverwendet werden können. Vereinfachen Sie den Testcode: Durch die Erfassung von Werten vereinfachen Abschlüsse den Testcode, indem sie die Notwendigkeit beseitigen, Parameter für jede Schleife wiederholt festzulegen. Verbessern Sie die Lesbarkeit: Verwenden Sie Abschlüsse, um die Testlogik zu organisieren und so den Testcode klarer und leichter lesbar zu machen.

Verkettete Aufrufe und Schließungen von PHP-Funktionen Verkettete Aufrufe und Schließungen von PHP-Funktionen Apr 13, 2024 am 11:18 AM

Ja, die Einfachheit und Lesbarkeit des Codes können durch verkettete Aufrufe und Abschlüsse optimiert werden: Verkettete Aufrufe verknüpfen Funktionsaufrufe in einer fließenden Schnittstelle. Abschlüsse erstellen wiederverwendbare Codeblöcke und greifen auf Variablen außerhalb von Funktionen zu.

Wie werden Schließungen in Java implementiert? Wie werden Schließungen in Java implementiert? May 03, 2024 pm 12:48 PM

Abschlüsse in Java ermöglichen es inneren Funktionen, auf äußere Bereichsvariablen zuzugreifen, selbst wenn die äußere Funktion beendet wurde. Durch anonyme innere Klassen implementiert, enthält die innere Klasse einen Verweis auf die äußere Klasse und hält die äußeren Variablen aktiv. Schließungen erhöhen die Codeflexibilität, Sie müssen sich jedoch des Risikos von Speicherverlusten bewusst sein, da Verweise auf externe Variablen durch anonyme innere Klassen diese Variablen am Leben halten.

See all articles