Maison >développement back-end >tutoriel php >Comprenez-vous vraiment le PHP actuel ?
Il y a quelque temps, le projet de l'entreprise a été mis à niveau de PHP5.3 vers PHP7. Maintenant, le projet commence à utiliser certaines des nouvelles syntaxes et fonctionnalités de PHP7. En repensant aux versions 5.4, 5.5 et 5.6 de PHP, je me sens un peu en manque de connaissances, j'ai donc décidé de lire "Modern PHP" pour compléter certains concepts
. Lecture de ce livre
1. Caractéristiques
Ne faites pas cela, c'est facile de semer la confusion
Suggérez une ligne d'utilisation :
<?php use Symfony\Component\HttpFoundation\Request, Symfony\Component\HttpFoundation\Response, Symfony\Component\HttpFoundation\Cookie;Utiliser plusieurs espaces de noms dans un seul fichier Vous pouvez le faire, mais cela viole
<?php use Symfony\Component\HttpFoundation\Request; use Symfony\Component\HttpFoundation\Response; use Symfony\Component\HttpFoundation\Cookie;"Un fichier définit une classe" Bonne pratique pour
.
Espace de noms global Si vous souhaitez utiliser la classe
<?php namespace Foo { //code } namespace Bar { //code }native de PHP, vous devez ajouter un symbole avant le nom de la classe
. S'il n'y a pas de symbole avant Exception, la classe Exception sera trouvée dans l'espace de noms MyApp
2 Utilisez l'interface
<?php namespace My\App; class Foo { public function doSomething() { $exception = new \Exception(); } }
Utilisez l'interface. plus flexible et peut être délégué à d'autres pour implémenter les détails. Les utilisateurs n'ont qu'à se soucier de l'interface et non de l'implémentation. Cela peut bien découpler le code et faciliter l'expansion
3. Traits. 🎜>
Les traits sont des implémentations partielles de classes qui peuvent être mélangées dans une ou plusieurs classes PHP existantesPourquoi utiliser des traitsPour donner un exemple précis, par exemple, il existe deux classes, Car et Phone, qui nécessitent toutes deux la fonctionnalité GPS. Afin de résoudre ce problème, la première réaction est de créer une classe parent, puis de laisser
HériterMais parce que évidemment. , cet ancêtre n'appartient pas à la hiérarchie d'héritage respective.
La deuxième réaction est de créer une interface GPS, de définir l'interface fonctionnelle GPS, puis de laisser les classes Car et Both Phone implémenter cette interface. tout en conservant la hiérarchie d'héritage naturelle. Cependant, cela nécessite des fonctions GPS en double dans les deux, ce qui n'est pas cohérent avec DRY (ne vous répétez pas La troisième réaction est de créer un trait qui implémente le principe). Fonction GPS, puis mélangez cette caractéristique dans les classes Voiture et Téléphone. Il peut réaliser des fonctions, n'affecte pas la structure d'héritage, ne répète pas l'implémentation et est parfait.
Créer et utiliser des traits
Créer des traits
Utiliser des traits
Générateur
<?php trait MyTrait{ //实现 }
Le générateur PHP est une nouvelle fonctionnalité introduite dans PHP5.5.0, que de nombreux développeurs PHP ne comprennent pas. Le générateur est un simple itérateur, mais le générateur n'est pas obligé d'implémenter l'
interface Iterator<?php class MyClass { use MyTrait; // 类的实现 }. Le générateur calcule et produit les valeurs à parcourir selon les besoins. Sans
Pour faire face à ce scénario, la méthode habituelle consiste d'abord à
<?php function makeRange($length) { for ($i = 0; $i < $length; $i++) { yield $i; } } foreach (makeRange(1000000) as $i) { echo $i, PHP_EOL; }lire tout le contenu du fichier
et à le placer dans le tableau, puis à effectuer le traitement et bientôt. Le problème avec ce type de traitement est le suivant :
Lorsque le fichier est très volumineux, il faut beaucoup de ressources mémoire pour le lire en une seule fois. Le générateur est le plus adapté à ce scénario car il occupe très peu de mémoire système<?php function getRows($file) { $handle = fopen($file, 'rb'); if ($handle === false) { throw new Exception(); } while (feof($handle) === false) { yield fgetcsv($handle); } } foreach (getRows('data.csv') as $row) { print_r($row); }.
5. FermetureThéoriquement, la fermeture et la fonction anonyme sont des concepts différents. Cependant, PHP le traite comme le même concept.
Fermeture simpleRemarque : La raison pour laquelle nous pouvons appeler $closurevariable
est que la valeur de cette variable est une fermeture, et le Closure PackageObject implémente la méthode magique invoquée()
<?php $closure = function ($name) { return sprintf('Hello %s', $name); } echo $closure("Beck"); // 输出 --> “Hello Beck”. Tant qu'il y a () après le
nom de la variable , PHP trouvera et appellera la méthode Invocation().
附加状态
使用use关键字可以把多个参数传入闭包,此时要像PHP函数或方法的参数一样,使用逗号分隔多个参数。
<?php function enclosePerson($name) { return function ($doCommand) use ($name) { return sprintf('%s, %s', $name, $doCommand); }; } // 把字符串“Clay”封装在闭包中 $clay = enclosePerson('Clay'); // 传入参数,调用闭包 echo $clay('get me sweet tea!'); // 输出 --> "Clay, get me sweet tea!"
使用bindTo()方法附加闭包的状态
PHP框架经常使用bindTo()方法把路由URL映射到匿名回调函数上,框架会把匿名函数绑定到应用对象上,这么做可以在这个匿名函数中使用$this关键字引用重要的应用对象。例子如下:
<?php class App { protected $routes = array(); protected $responseStatus = '200 OK'; protected $responseContentType = 'text/html'; protected $responseBody = 'Hello world'; public function addRoute($routePath, $routeCallback) { $this->routes[$routePath] = $routeCallback->bindTo($this, CLASS);//重点 } public function dispatch($currentPath) { foreach ($this->routes as $routePath => $callback) { if ($routePath === $currentPath) { $callback(); } } header('HTTP/1.1' . $this->responseStatus); header('Content-type:' . $this->responseContentType); header('Content-length' . mb_strlen($this->responseBody)); echo $this->responseBody; } }
第11行是重点所在,把路由回调绑定到了当前的App实例上。这么做能在回调函数中处理App实例的状态:
<?php $app = new App(); $app->addRoute('/users/josh', function () { $this->responseContentType = 'application/json;charset=utf8'; $this->responseBody = '{"name": "Josh"}'; }); $app->dispatch('/users/josh');
字节码缓存不是PHP的新特性,很多独立的扩展可以实现缓存。从PHP5.5.0开始,PHP内置了字节码缓存功能,名为Zend OPcache。
字节码缓存是什么
PHP是解释性语言,PHP解释器执行PHP脚本时会解析PHP脚本代码,把PHP代码编译成一系列Zend操作码,然后执行字节码。每次请求PHP文件都是这样,会消耗很多资源。字节码缓存能存储预先编译好的PHP字节码。这意味着,请求PHP脚本时,PHP解释器不用每次都读取、解析和编译PHP代码。这样能极大地提升应用的性能。
从PHP5.4.0起,PHP内置了Web服务器,这对众多使用Apache或nginx的php开发者来说,可能是个隐藏功能。不过,这个内置的服务器功能并不完善,不应该在生产环境中使用,但对本地开发来说是个便利的工具,可以用于快速预览一些框架和应用。
启动服务器
php -S localhost:4000
php -S localhost:8000 -c app/config/php.ini
路由器脚本
与Apache和nginx不同,它不支持.htaccess文件。因此,这个服务器很难使用多数流行的PHP框架中常见的前端控制器。PHP内置的服务器使用路由器脚本弥补了这个遗漏的功能。处理每个HTTP请求前,会先经过这个路由器脚本,如果结果为false,返回当前HTTP请求中引用的静态资源URI。
php -S localhost:8000 route.php
是否为内置的服务器
<?php if (php_sapi_name() === 'cli-server') { // php 内置的web服务器 }
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!