你真的了解php吗?

一、特性

1. 命名空间

命名空间用的比较多,不详细写了,记录几个值得注意的实践和细节。 多重导入 别这么做,这样写容易让人困惑。

<?php
use Symfony\HttpFoundation\Request,
    Symfony\HttpFoundation\Rese,
    Symfony\HttpFoundation\Cookie;

建议一行写一个use语句:

<?php
use Symfony\HttpFoundation\Request;
use Symfony\HttpFoundation\Rese;
use Symfony\HttpFoundation\Cookie;

一个文件中使用多个命名空间 你可以这么做,但这违背了“一个文件定义一个类”的良好实践。

<?php
namespace Foo {  
//code
}namespace Bar {  
//code 
}

全局命名空间 想要使用PHP原生的Exception类,需要在类名前加 \ 符号。

<?php
namespace My\App;
class Foo{  
public function doSomething()
  {
    $exception = new \Exception();
  }
}

如果Exception前不加 \ 符号,会在My\App命名空间下寻找Exception类。

2. 使用接口

使用接口编写的代码更灵活,能委托其他人实现细节。使用的人只需要关心有什么接口,而不需要关心实现。能够很好地解耦代码,方便扩展,比较常用就不说啦。

3. 性状

在学习laravel框架之前都没弄清楚性状(trait)。这是PHP5.4.0引入的新概念,既像类又像接口。但它两个都不是。

性状是类的部分实现,可以混入一个或多个现有PHP类中。类似Ruby的组合模块活混入(mixin)。

为什么使用性状 举个具体的例子,比如有两个类,Car 和 Phone,他们都需要GPS功能。为了解决这个问题,第一反应创建一个父类,然后让Car和Phone继承它。但因为很明显,这个祖先不属于各自的继承层次结构。

第二反应创建一个GPS的接口,定义好GPS的功能接口,然后让Car和Phone两个类都实现这个接口。这样做能实现功能,同时也能保持自然的继承层级结构。不过,这就使得在两个都要实现重复的GPS功能,这不符合DRY(dont repeat yourself)原则。

第三反应创建实现GPS功能的性状(trait),然后在Car和Phone类中混入这个性状。能实现功能,不影响继承结构,不重复实现,完美。

创建与使用性状 创建trait

<?phptrait MyTrait{ 
//实现
}

使用trait

<?php
class MyClass{  
use MyTrait;  
// 类的实现
}

4. 生成器

PHP生成器(generator)是PHP5.5.0引入的新功能,很多PHP开发者生成器不了解。生成器是个简单的迭代器,但生成器不要求实现Iterator接口。生成器会根据需要计算并产生要迭代的值。如果不查询,生成器永远不知道下一个要迭代的值是什么,在生成器中无法后退或快进。具体看如下两个例子:

简单的生成器

<?php
function makeRange($length) {  
  for ($i = 0; $i < $length; $i++) {    
    yield $i;
  }
}
foreach (makeRange(1000000) as $i) {  
echo $i, PHP_EOL;
}

具体场景:使用生成器处理CSV文件

<?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. 闭包

理论上,闭包和匿名函数是不同的概念。不过,PHP将其视作相同的概念。 简单闭包

<?php
$closure = function ($name) {  
    return sprintf('Hello %s', $name);
}
echo $closure("Beck");
// 输出 --> “Hello Beck”

注意:我们之所以能调用$closure变量,是因为这个变量的值是个闭包,而且闭包对象实现了__invoke()魔术方法。只要变量名后有(),PHP就会查找并调用__invoke()方法。

附加状态 使用use关键字可以把多个参数传入闭包,此时要像PHP函数或方法的参数一样,使用逗号分隔多个参数。

<?php
function enclosePerson($name) {  
 return function ($doCom) use ($name) {    
    return sprintf('%s, %s', $name, $doCom);
  };
}// 把字符串“Clay”封装在闭包中
$clay = enclosePerson('Clay');
// 传入参数,调用闭包echo $clay('get tea!');
// 输出 --> "Clay, get tea!"

6. Zend OPcache

字节码缓存不是PHP的新特性,很多独立的扩展可以实现缓存。从PHP5.5.0开始,PHP内置了字节码缓存功能,名为Zend OPcache。

字节码缓存是什么 PHP是解释性语言,PHP解释器执行PHP脚本时会解析PHP脚本代码,把PHP代码编译成一系列Zend操作码,然后执行字节码。每次请求PHP文件都是这样,会消耗很多资源。字节码缓存能存储预先编译好的PHP字节码。这意味着,请求PHP脚本时,PHP解释器不用每次都读取、解析和编译PHP代码。这样能极大地提升应用的性能。

7. 内置的HTTP服务器

从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服务器
}

原文发布于微信公众号 - php(phpdaily)

原文发表时间:2017-02-04

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏码洞

自己动手实现 Shell多进程套套符

一篇技术文章如今仅仅是理论上讲得天花乱坠,却不能自己撸出东西来,那么它写的再好,也只能算纸上谈兵。继上一篇 《我们天天都在使用的套套符命令,Shell 在里面到...

691
来自专栏耕耘实录

Linux中find命令的使用详解(中)

版权声明:本文为耕耘实录原创文章,各大自媒体平台同步更新。欢迎转载,转载请注明出处,谢谢

792
来自专栏自由而无用的灵魂的碎碎念

解决source insight 3.5遇到的parse too comples错误

我使用source insight 编写c,编写代码的时候,source insight的symbol windows可能会提示parse too comple...

972
来自专栏Android随笔

mac学习笔记

在打开的文本文件中,添加如下内容(/XXX/XXX/platform-tools/为你的adb路径)

692
来自专栏Java3y

看完这篇Linux基本的操作就会了

2746
来自专栏nummy

sphinx入门指南【1】快速入门

包含.rst文件的根目录称之为源文件目录,目录中还包含sphinx的配置文件conf.py。

1194
来自专栏python3

python 文件操作2

Somehow, it seems the love I knew was always the most destructive kind

861
来自专栏Python攻城狮

Python操作redis1.python操作redis2.实例3.Redis缓存Mysql模拟用户登录,Python实现实例

1 .用户登录首先判断是否在redis缓存中,如果在redis缓存中,直接登录成功; 2 .若用户未在redis缓存,则访问Mysql,判...

1123
来自专栏挖坑填坑

ABP框架学习之—— 设置管理

abp这个设置管理时基于服务器缓存实现,使用得时string类型得键值对(建值都是string类型)。通过设置器(SettingProvider)注入,在站点启...

783
来自专栏Java后端技术

一段奇妙的vim编辑器之旅

  对于Linux服务器上的操作,我们往往少不了使用vim,而有时候我对vim的使用并没有那么的熟练和深入,这周就深入的学习了vim的使用,包括入门和进阶,先分...

733

扫码关注云+社区