Maison >développement back-end >tutoriel php >Description de l'application pratique des fonctionnalités de fermeture PHP

Description de l'application pratique des fonctionnalités de fermeture PHP

伊谢尔伦
伊谢尔伦original
2017-07-01 13:12:531846parcourir

La nouvelle version de PHP suit de nombreuses nouvelles fonctionnalités, l'une des fonctionnalités les plus accrocheuses est la prise en charge des fermetures. Alors à l'avenir, pourrons-nous écrire du code très sympa comme ces gars qui écrivent Ruby, Javascript et d'autres "langages high-tech" ?

Eh bien, dans la plupart des cas, c’est possible, mais il y a quand même certains aspects qui sont très troublants. Parlons-en lentement.
De nombreuses langues multiples fournissent des méthodes très élégantes et belles pour faire fonctionner les tableaux. Dans l'exemple suivant, la fonction de fermeture fournie par PHP5.3 et d'autres langages sera utilisée pour montrer comment faire fonctionner "objectivement" le tableau itéré.
Note de traduction : l'auteur de l'article original est relativement incompétent. Je ne connais pas les langages Groovy et Scala, je vais donc ajouter l'implémentation Javascript ici.
Avant de commencer, laissez-moi vous expliquer que cet exemple est juste pour illustrer le propos et ne prend pas en compte d'autres facteurs tels que les performances.

« Shop around »

Commencez par un exemple simple, avec le tableau suivant :
$nums = array(10, 20, 30, 40 ); Besoin de trouver des éléments supérieurs à 15 dans le tableau. Ensuite, sans considérer la fermeture, nous pourrions écrire comme ceci :
$res = array();foreach ($nums as $n) { if ($n > 15) { $res[] = $ n }} Si le langage lui-même a support de fermeture , alors il peut être écrit comme ceci (langage Groovy)
def res = nums.findAll { it > 15 } ou utiliser le langage Scala
val res = nums filter (_ > 15) Remarque de traduction : pour Javascript 1.6, ce sera comme suit
var res = nums.filter(function(c){return c > 15}); , Vous pouvez donc voir que Groovy, Scala (et Javascript) sont tous très beaux et peuvent être réalisés sur une seule ligne.
Bien sûr, si vous utilisez les fermetures PHP5.3, vous pouvez également le faire
$res = array_filter($nums, function($v) { return $v > 15; } ) ;PHP utilise plus de caractères que Scala pour cela, mais il est plus court et plus facile à lire que l'exemple précédent.

À propos, le code PHP ci-dessus utilise en fait l'analyse Lambda et n'est pas une véritable fermeture. Ce n'est pas l'objet de notre attention actuelle. Pour des informations détaillées sur les fermetures PHP et l'analyse Lambda, veuillez vous référer ici.
Jusqu'à présent, cela semble plutôt bien, alors augmentons la difficulté du problème : trouvez tous les éléments supérieurs à 15, puis multipliez par 2 plus une certaine valeur variable dans la portée, puis revenez.

Implémentation Groovy :
def x = 1def res = nums .findAll { it > 15 } .collect { it * 2 + x }Implémentation Scala :
val x = 1val res = nums filter (_ > 15) annotation de traduction map (_ * 2 + x), implémentation Javascript :
var i = 1;var res = nums.filter(function(c){return c > 15}). function(c){return c * 2 + i}); et PHP :
$x = 1;$res = array_map( function($v) use ($x) { return $v * 2 + $x; }, array_filter( $nums, function($v) { return $v > 15; })); En termes de taille de code, il semble que PHP soit différent des autres langages. Laissant de côté l'esthétique littérale du code, il existe un problème supplémentaire avec le code PHP ci-dessus.
Par exemple, que se passe-t-il si vous devez utiliser des clés de tableau au lieu de valeurs à des fins de comparaison ? Oui, le code ci-dessus ne peut pas être réalisé. De plus, syntaxiquement parlant, le code ci-dessus est très difficile à lire.

Retour à la nature, il faut encore revenir à l'ancienne façon de penser pour résoudre le problème :
$x = 1;$res = array();foreach ($nums as $n ) { if ($n > ; 15) { $res[] = $n * 2 + $x }} Ouf, cela semble encore une fois clair. Mais à ce moment-là, vous pourriez être à nouveau confus : « Pourquoi s'embêter ? N'est-ce pas juste une opération de tableau ?
Oui, le meilleur est à venir. À l’heure actuelle, il est temps de laisser certaines des fonctionnalités avancées de PHP entrer en jeu pour résoudre ce « problème ennuyeux » qui semble avoir des tendances autodestructrices.

ArrayObject – une encapsulation de tableaux
PHP possède une bibliothèque standard appelée SPL, qui contient une classe appelée ArrayObject, qui peut fournir la fonction de "classes opérationnelles comme des tableaux", telles que
$res = new ArrayObject(array(10, 20, 30, 40));foreach ($res as $v) { echo "$vn";}ArrayObject est une classe intégrée, vous pouvez donc l'utiliser comme les autres classes L'encapsuler .

Arr - enrobage de sucre

Maintenant que nous avons les fonctionnalités d'ArrayObject et les fermetures, nous pouvons commencer à essayer de l'encapsuler :
class Arr extends ArrayObject { static function make($array) { return new self($array); } function map($func) { $res = new self(); foreach ($this as $k => $v) { $res[ $k ] = $func($k, $v); } return $res; } function filter($func) { $res = new self(); foreach ($this as $k => $v) { if ($ func($k, $v)) { $res[$k] = $v; } } return $res }}D'accord, tout est prêt. Le code PHP réécrit ci-dessous peut résoudre les problèmes mentionnés ci-dessus, et ressemble « presque » syntaxiquement :

$res = Arr::make($nums) ->filter(function($k, $v) { return $v > 15; }) ->map(function($k, $v) { return $ v * 2; });En quoi le code ci-dessus est-il différent de la méthode traditionnelle ? Premièrement, ils peuvent être des appels récursifs et en chaîne, de sorte que davantage d’opérations similaires peuvent être ajoutées.

En même temps, les clés et les valeurs du tableau peuvent être manipulées respectivement via les deux paramètres du rappel - $k correspond à la clé et $v correspond à la valeur. Cela nous permet d'utiliser des valeurs clés dans les fermetures, ce qui n'est pas possible avec la fonction PHP traditionnelle array_filter.
Un autre avantage supplémentaire réside dans les appels API plus cohérents. En utilisant les opérations de fonctions PHP traditionnelles, leur premier paramètre peut être une fermeture, ou un tableau, ou plusieurs tableaux... Quoi qu'il en soit, qui sait ?
Voici le code source complet de la classe Arr, qui contient également d'autres fonctions utiles (similaires à réduire et marcher. En fait, leur implémentation est similaire au code).

Jeu

Cette question est en fait difficile à répondre - elle dépend de nombreux facteurs tels que le contexte du code et le programmeur lui-même. En fait, lorsque j'ai vu pour la première fois l'implémentation des fermetures dans PHP, j'ai eu l'impression d'être de retour à l'époque de Java il y a longtemps, lorsque j'ai commencé à utiliser des classes internes anonymes pour implémenter les fermetures. Bien sûr, cela peut être fait, mais cela semble superflu. Il n'y a rien de mal avec les fermetures PHP, mais leur implémentation et leur syntaxe me déroutent.
D'autres langages dotés de fonctionnalités de fermeture, ils peuvent appeler des fermetures très facilement et ont une syntaxe élégante. Dans l'exemple ci-dessus, l'utilisation d'une boucle traditionnelle dans Scala fonctionnerait également, mais l'écririez-vous de cette façon ? D'un autre côté, certaines personnes disent que le problème ci-dessus peut également être résolu en utilisant des fermetures PHP, mais l'écririez-vous généralement comme ceci ?
Il est certain que les fermetures PHP peuvent être un couteau tranchant dans certaines situations (comme les retards d'exécution et les appels de ressources), mais elles sont un peu difficiles face aux opérations traditionnelles d'itération et de tableau. Ne vous découragez pas. Quoi qu’il en soit, le plus important est de revenir à l’essentiel et d’écrire du code et des API compatibles et propres.

Conclusion

Comme toutes les fonctionnalités de syntaxe qui ont été ajoutées plus tard (vous vous souvenez de la fonctionnalité Generics de Java à l'époque ? Et de la fonctionnalité PHP OOP des années précédentes), elles nécessitent toutes Il est temps de s’introduire et éventuellement de se stabiliser. À mesure que PHP5.3 et même PHP6 deviendront plus populaires à l'avenir, je pense que de plus en plus de techniques et de fonctionnalités seront progressivement découvertes par les programmeurs intelligents dans un avenir proche.
Retournez au titre au début de l'article original et comparez
$res = Arr::make($nums) ->filter(function($k, $v) { return $v > 15; }) - >map(function($k, $v) { return $v * 2; }); et
val res = nums filter (_ > 15) map (_ * 2). En dernière analyse, il ne s’agit que de syntaxe. Essentiellement, ils résolvent le même problème par des approches différentes. Les caractéristiques d'application des langages de programmation sont différentes, il est donc impossible de comparer lequel est le meilleur et lequel est le pire.
Enfin, voici les exemples de code de cet article. Je pense que vous pouvez trouver plus d'informations sur la façon d'utiliser PHP pour l'itération fonctionnelle (bien sûr pas seulement ceux-ci).


Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
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