Maison interface Web js tutoriel Comprendre les fermetures dans les compétences de programmation fonctionnelle javascript_javascript

Comprendre les fermetures dans les compétences de programmation fonctionnelle javascript_javascript

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

La fermeture est un concept de programmation fonctionnelle apparu dans les années 1960. Le premier langage à implémenter la fermeture était Scheme, qui est un dialecte de LISP. Depuis lors, les fonctionnalités de fermeture ont été largement adoptées par d’autres langages.
La définition stricte d'une fermeture est "un ensemble composé d'une fonction (environnement) et de ses variables libres incluses". Cette définition est un peu obscure pour tout le monde, alors expliquons-la d'abord à travers des exemples et une explication moins stricte est une fermeture, puis donne des exemples de quelques utilisations classiques des fermetures.

Qu'est-ce que la fermeture

En termes simples, chaque fonction en JavaScript est une fermeture, mais en général, les fonctions imbriquées peuvent mieux refléter
Pour montrer les caractéristiques de fermeture, veuillez consulter l'exemple suivant :

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
Copier après la connexion

Dans ce code, il y a un nombre de variables locales dans la fonction generateClosure(), avec une valeur initiale de 0. Il existe également une fonction appelée get, qui augmente la variable count dans sa portée parent, la fonction generateClosure(), de 1 et renvoie la valeur de count. La valeur de retour de generateClosure() est la fonction get. En externe, nous avons appelé la fonction generateClosure() via la variable counter et obtenu sa valeur de retour, qui est la fonction get. Ensuite, nous avons appelé à plusieurs reprises counter() et nous avons constaté que la valeur renvoyée était incrémentée de 1 à chaque fois.
Jetons un coup d'œil aux caractéristiques de l'exemple ci-dessus. Selon la compréhension habituelle de la programmation impérative, count est une variable à l'intérieur de la fonction generateClosure. Son cycle de vie est la période pendant laquelle generateClosure est appelé. la variable count L'espace appliqué sera libéré. Le problème est qu'après la fin de l'appel à generateClosure(), counter() fait référence à la variable count "déjà publiée", et non seulement aucune erreur ne se produit, mais count est modifié et renvoyé à chaque fois que counter() est appelé. Que se passe-t-il?
C’est exactement la caractéristique de ce qu’on appelle la fermeture. Lorsqu'une fonction renvoie une fonction définie en son sein, une fermeture est générée. La fermeture inclut non seulement la fonction renvoyée, mais également l'environnement dans lequel la fonction est définie. Dans l'exemple ci-dessus, lorsque la fonction interne get de la fonction generateClosure() est référencée par une variable externe counter, les variables locales de counter et generateClosure() sont une fermeture. Si ce n'est pas assez clair, l'exemple suivant peut vous aider
Vous comprenez :

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
Copier après la connexion

L'exemple ci-dessus explique comment les fermetures sont générées : counter1 et counter2 appellent respectivement la fonction generateClosure(), générant deux instances de fermetures, et les variables de comptage qu'ils référencent en interne appartiennent à leurs environnements d'exploitation respectifs. Nous pouvons comprendre que lorsque generateClosure() renvoie la fonction get, les variables internes de la fonction generateClosure() auxquelles get peut faire référence (c'est-à-dire la variable count) sont également renvoyées en privé, et une copie est générée dans la mémoire, et then generateClosure( ) Les deux instances de la fonction renvoyée, counter1 et counter2, sont indépendantes l'une de l'autre.

Le but de la fermeture

1. Fonction de rappel imbriquée
Les fermetures ont deux utilisations principales, l'une consiste à implémenter des fonctions de rappel imbriquées et l'autre à masquer les détails d'un objet. Examinons d'abord l'exemple de code suivant pour comprendre les fonctions de rappel imbriquées. Le code suivant utilise MongoDB dans Node.js pour implémenter une fonction simple d'ajout d'utilisateurs :

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);
});
}
});
});
});
});
});
};
Copier après la connexion

Si vous n'êtes pas familier avec Node.js ou MongoDB, cela n'a pas d'importance. Vous n'avez pas besoin de comprendre les détails, comprenez simplement la logique générale. Ce code utilise des couches d'imbrication de fermetures, et chaque couche d'imbrication est une fonction de rappel. La fonction de rappel ne sera pas exécutée immédiatement, mais sera rappelée par la fonction demandée après le traitement de la requête correspondante. Nous pouvons voir qu'il existe une référence au rappel dans chaque couche d'imbrication, et la couche la plus interne utilise également la variable uid définie par la couche externe. En raison de l'existence du mécanisme de fermeture, même si la fonction externe a été exécutée, les variables appliquées dans sa portée ne seront pas libérées, car la fonction interne peut toujours référencer ces variables, réalisant ainsi parfaitement le rappel asynchrone imbriqué.

2. Implémenter les membres privés
Nous savons que les objets JavaScript n'ont pas de propriétés privées, ce qui signifie que chaque propriété de l'objet est exposée au monde extérieur. Cela peut engendrer des risques de sécurité, par exemple si l'utilisateur de l'objet modifie directement un attribut, entraînant la destruction de la cohérence des données internes de l'objet, etc. JavaScript utilise la convention pour placer un trait de soulignement avant toutes les propriétés privées (telles que _myPrivateProp) pour indiquer que cette propriété est privée et que les objets externes ne doivent pas la lire ou l'écrire directement. Mais il ne s’agit que d’un accord informel. En supposant que l’utilisateur de l’objet ne le fasse pas, existe-t-il un mécanisme plus strict ? La réponse est oui, cela peut être réalisé grâce à des fermetures. Reprenons l'exemple précédent :

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
Copier après la connexion

Kita dapat melihat bahawa hanya memanggil counter() boleh mengakses pembolehubah kiraan dalam penutupan dan menambahnya sebanyak 1 mengikut peraturan Tidak mustahil untuk mencari pembolehubah kiraan dengan cara lain. Diilhamkan oleh contoh mudah ini, kita boleh merangkum objek dengan penutupan dan hanya mengembalikan objek "aksesor" untuk menyembunyikan butiran.

Di atas ialah keseluruhan kandungan artikel ini, saya harap ia dapat membantu semua orang mempelajari dan memahami penutupan JavaScript dengan lebih baik.

Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn

Outils d'IA chauds

Undresser.AI Undress

Undresser.AI Undress

Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover

AI Clothes Remover

Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool

Undress AI Tool

Images de déshabillage gratuites

Clothoff.io

Clothoff.io

Dissolvant de vêtements AI

Video Face Swap

Video Face Swap

Échangez les visages dans n'importe quelle vidéo sans effort grâce à notre outil d'échange de visage AI entièrement gratuit !

Outils chauds

Bloc-notes++7.3.1

Bloc-notes++7.3.1

Éditeur de code facile à utiliser et gratuit

SublimeText3 version chinoise

SublimeText3 version chinoise

Version chinoise, très simple à utiliser

Envoyer Studio 13.0.1

Envoyer Studio 13.0.1

Puissant environnement de développement intégré PHP

Dreamweaver CS6

Dreamweaver CS6

Outils de développement Web visuel

SublimeText3 version Mac

SublimeText3 version Mac

Logiciel d'édition de code au niveau de Dieu (SublimeText3)

Quelle est la signification de la fermeture dans l'expression lambda C++ ? Quelle est la signification de la fermeture dans l'expression lambda C++ ? Apr 17, 2024 pm 06:15 PM

En C++, une fermeture est une expression lambda qui peut accéder à des variables externes. Pour créer une fermeture, capturez la variable externe dans l'expression lambda. Les fermetures offrent des avantages tels que la réutilisabilité, la dissimulation des informations et une évaluation paresseuse. Ils sont utiles dans des situations réelles telles que les gestionnaires d'événements, où la fermeture peut toujours accéder aux variables externes même si elles sont détruites.

Comment implémenter la fermeture dans une expression C++ Lambda ? Comment implémenter la fermeture dans une expression C++ Lambda ? Jun 01, 2024 pm 05:50 PM

Les expressions C++ Lambda prennent en charge les fermetures, qui enregistrent les variables de portée de fonction et les rendent accessibles aux fonctions. La syntaxe est [capture-list](parameters)->return-type{function-body}. capture-list définit les variables à capturer. Vous pouvez utiliser [=] pour capturer toutes les variables locales par valeur, [&] pour capturer toutes les variables locales par référence, ou [variable1, variable2,...] pour capturer des variables spécifiques. Les expressions Lambda ne peuvent accéder qu'aux variables capturées mais ne peuvent pas modifier la valeur d'origine.

Quels sont les avantages et les inconvénients des fermetures dans les fonctions C++ ? Quels sont les avantages et les inconvénients des fermetures dans les fonctions C++ ? Apr 25, 2024 pm 01:33 PM

Une fermeture est une fonction imbriquée qui peut accéder aux variables dans la portée de la fonction externe. Ses avantages incluent l'encapsulation des données, la conservation de l'état et la flexibilité. Les inconvénients incluent la consommation de mémoire, l’impact sur les performances et la complexité du débogage. De plus, les fermetures peuvent créer des fonctions anonymes et les transmettre à d'autres fonctions sous forme de rappels ou d'arguments.

Résoudre le problème de fuite de mémoire causé par les fermetures Résoudre le problème de fuite de mémoire causé par les fermetures Feb 18, 2024 pm 03:20 PM

Titre : Fuites de mémoire causées par les fermetures et solutions Introduction : Les fermetures sont un concept très courant en JavaScript, qui permettent aux fonctions internes d'accéder aux variables des fonctions externes. Cependant, les fermetures peuvent provoquer des fuites de mémoire si elles ne sont pas utilisées correctement. Cet article explorera le problème de fuite de mémoire provoqué par les fermetures et fournira des solutions et des exemples de code spécifiques. 1. Fuites de mémoire causées par les fermetures La caractéristique des fermetures est que les fonctions internes peuvent accéder aux variables des fonctions externes, ce qui signifie que les variables référencées dans les fermetures ne seront pas récupérées. S'il est mal utilisé,

Appels enchaînés et fermetures de fonctions PHP Appels enchaînés et fermetures de fonctions PHP Apr 13, 2024 am 11:18 AM

Oui, la simplicité et la lisibilité du code peuvent être optimisées grâce à des appels et des fermetures enchaînés : les appels en chaîne lient les appels de fonction dans une interface fluide. Les fermetures créent des blocs de code réutilisables et accèdent à des variables en dehors des fonctions.

L'impact des pointeurs de fonction et des fermetures sur les performances de Golang L'impact des pointeurs de fonction et des fermetures sur les performances de Golang Apr 15, 2024 am 10:36 AM

L'impact des pointeurs de fonction et des fermetures sur les performances de Go est le suivant : Pointeurs de fonction : légèrement plus lents que les appels directs, mais améliorent la lisibilité et la réutilisabilité. Fermetures : généralement plus lentes, mais encapsulent les données et le comportement. Cas pratique : les pointeurs de fonction peuvent optimiser les algorithmes de tri et les fermetures peuvent créer des gestionnaires d'événements, mais ils entraîneront des pertes de performances.

Le rôle de la fermeture de la fonction Golang dans les tests Le rôle de la fermeture de la fonction Golang dans les tests Apr 24, 2024 am 08:54 AM

Les fermetures de fonctions du langage Go jouent un rôle essentiel dans les tests unitaires : Capture de valeurs : les fermetures peuvent accéder aux variables dans la portée externe, permettant ainsi de capturer et de réutiliser les paramètres de test dans des fonctions imbriquées. Simplifiez le code de test : en capturant les valeurs, les fermetures simplifient le code de test en éliminant le besoin de définir des paramètres à plusieurs reprises pour chaque boucle. Améliorez la lisibilité : utilisez des fermetures pour organiser la logique de test, rendant ainsi le code de test plus clair et plus facile à lire.

Comment les fermetures sont-elles implémentées en Java ? Comment les fermetures sont-elles implémentées en Java ? May 03, 2024 pm 12:48 PM

Les fermetures en Java permettent aux fonctions internes d'accéder aux variables de portée externe même si la fonction externe est terminée. Implémentée via des classes internes anonymes, la classe interne contient une référence à la classe externe et maintient les variables externes actives. Les fermetures augmentent la flexibilité du code, mais vous devez être conscient du risque de fuite de mémoire, car les références à des variables externes par des classes internes anonymes maintiennent ces variables en vie.

See all articles