PHP发展这么多年,技术、架构都已经革新,了解现代PHP很重要,最近在看Model PHP这本书,系统的了解下PHP相关的概念。

性状 Trait

是类的部分实现(即常量、属性和方法),可以混入一个或多个现有的php类中。

性状有两个作用:表明类可以做什么(接口);提供模块化实现(像是类)。

比如说两个无关的类需要拥有一个共同的方法,继承、接口都不太合理(一是属性不同;二是代码重复),使用性状可以共同使用某个方法。

举例 汽车和快递员都可能都需要查询地理位置

  1. trait Geocodable{
  2. protected $address;
  3. protected $geocoder
  4. //获取经纬度
  5. public function getLatitude(){
  6. }
  7. }

使用性状:

  1. <?php
  2. Class Car{
  3. use Geocodable;
  4. //类的实现
  5. ...
  6. }

生成器

生成器提供了一种更容易的方法来实现简单的对象迭代,相比较定义类实现 Iterator 接口(正常迭代)的方式,性能开销和复杂性大大降低。

生成器允许你在 foreach 代码块中写代码来迭代一组数据而不需要在内存中创建一个数组, 那会使你的内存达到上限,或者会占据可观的处理时间。相反,你可以写一个生成器函数,就像一个普通的自定义函数一样, 和普通函数只返回一次不同的是, 生成器可以根据需要 yield 多次,以便生成需要迭代的值。

生成器根据需求计算出要迭代的值;及时计算并产出后续值,不占用宝贵的内存资源。

一个简单的例子就是使用生成器来重新实现 range() 函数。 标准的 range() 函数需要在内存中生成一个数组包含每一个在它范围内的值,然后返回该数组, 结果就是会产生多个很大的数组。 比如,调用 range(0, 1000000) 将导致内存占用超过 100 MB。

做为一种替代方法, 我们可以实现一个 xrange() 生成器, 只需要足够的内存来创建 Iterator 对象并在内部跟踪生成器的当前状态,这样只需要不到1K字节的内存。

Example #1 将 range() 实现为生成器

  1. <?php
  2. function xrange($start, $limit, $step = 1) {
  3. if ($start < $limit) {
  4. if ($step <= 0) {
  5. throw new LogicException('Step must be +ve');
  6. }
  7. for ($i = $start; $i <= $limit; $i += $step) {
  8. yield $i;
  9. }
  10. } else {
  11. if ($step >= 0) {
  12. throw new LogicException('Step must be -ve');
  13. }
  14. for ($i = $start; $i >= $limit; $i += $step) {
  15. yield $i;
  16. }
  17. }
  18. }
  19. /*
  20. * 注意下面range()和xrange()输出的结果是一样的。
  21. */
  22. echo 'Single digit odd numbers from range(): ';
  23. foreach (range(1, 9, 2) as $number) {
  24. echo "$number ";
  25. }
  26. echo "\n";
  27. echo 'Single digit odd numbers from xrange(): ';
  28. foreach (xrange(1, 9, 2) as $number) {
  29. echo "$number ";
  30. }
  31. ?>

以上例程会输出:

  1. Single digit odd numbers from range(): 1 3 5 7 9
  2. Single digit odd numbers from xrange(): 1 3 5 7 9

想象下处理斐波那契数列或者迭代流资源,假设处理一个4GB的csv文件,而虚拟服务器只允许使用1GB内存,如果整个加载到内存中,服务器就崩了,而用生成器一次只会为csv一行分配内存。

  1. <?php
  2. function getRows( $file ){
  3. $handle = fopen($file,'rb');
  4. if( $handle === false ){
  5. throw new Exception();
  6. }
  7. while(feof($handle) === false){
  8. yield fgetcsv($handle);
  9. }
  10. fclose($handle);
  11. }
  12. foreach($getRows('data.csv') as $row){
  13. print_r($row);
  14. }

生成器是功能多样性和简洁性之间的这种方案。生成器是只能前进的迭代器,无法执行快进,后退,查找操作。

参考鸟哥的文章:http://www.laruence.com/2015/05/28/3038.html

闭包

是指在创建时封装周围状态的函数。

匿名函数其实就是没有名称的函数,特别适合作为函数或方法的调用。

  • _invoke 尝试以函数方式调用一个对象时,该方法自动被调用。

创建简单的闭包

  1. $closure = function($name){
  2. return sprintf ('hello %s',$name);
  3. }
  4. echo $closure("Josh");
  5. //输出 --> hello Josh

作为回调

  1. //匿名回调
  2. $numbersPlusOne = array_map(function( $number ){
  3. return $number+1;
  4. },[1,2,3]);
  5. print_r($numbersPlusOne);
  6. //输出 --> [2,3,4]
  1. //具名回调
  2. function incrementNumber($number){
  3. return $number+1;
  4. }
  5. $numbersPlusone = array_map('incrementNumber',[1,2,3]);
  6. print_r($numbersPlusOne);
  7. //输出 --> [2,3,4]
  • 附加状态

PHP闭包附加并封装状态,手动调用闭包对象的bindTo()方法或者使用use关键字,把状态附加到PHP闭包上。

实例:

  1. <?php
  2. function enclosePersion( $name ){
  3. return function($doCommand use $name){
  4. return sprintf('%s,%s',$name,$doCommand);
  5. }
  6. }
  7. //将字符串“sun”封装到闭包中
  8. $sun = enclosePersion( 'sun' );
  9. //传入参数,调用闭包
  10. echo $sun('get me sweet tea!');
  11. //输出 --> "sun,get me sweet tea!"

PHP闭包是对象,与任何其他PHP对象类似,每个闭包实例都可以使用$this关键字获取闭包的内部状态。闭包有__invoke()、bindTo()方法;

  • [ ] 参考下手册,手册有详细说明bindTo、bind;

  1. <?php
  2. class A {
  3. function __construct ( $val ) {
  4. $this -> val = $val ;
  5. }
  6. function getClosure () {
  7. //returns closure bound to this object and scope
  8. return function() { return $this -> val ; };
  9. }
  10. }
  11. $ob1 = new A ( 1 );
  12. $ob2 = new A ( 2 );
  13. $cl = $ob1 -> getClosure ();
  14. echo $cl (), "\n" ;
  15. $cl = $cl -> bindTo ( $ob2 );
  16. echo $cl (), "\n" ;
  17. ?>
  18. 以上例程的输出类似于:
  19. 1
  20. 2

Zend OPcache

从php 5.5.0开始,内置了字节码缓存功能。

PHP执行过程:

  1. graph LR
  2. 解析脚本代码-->编译成一系列Zend操作码
  3. 编译成一系列Zend操作码-->执行字节码

字节码缓存 存储好预先编译好的PHP字节码,PHP解释器跳过读取、解析、编译PHP代码,直接从内存中读取预先编译好的字节码,然后执行。优点节省了时间,极大地提升了应用的性能。

脚本执行图解:

  • 启用Zend OPcache

编译PHP时,明确指定启用Zend OPcache。

执行./configure 命令时必须包含以下选项:

  1. --enable-opcache

编译成功会显示Zend OPcache扩展的文件路径。也可通过下述命令查找扩展的安装路径

  1. php-config --extension-dir

php.ini文件中指定Zend OPcache扩展路径,如:

  1. zend_extension=/path/to/opcache.so

更新php.ini文件,重启php进程。

  1. <?php
  2. phpinfo()

查看Zend OPcache扩展开启情况

  • 配置Zend OPcache
  1. [opcache]
  2. zend_extension=/usr/local/php5/lib/php/extensions/no-debug-non-zts-20131226/opcache.so
  3. ; Zend Optimizer + 的开关, 关闭时代码不再优化.
  4. opcache.enable=1
  5. ; Determines if Zend OPCache is enabled for the CLI version of PHP
  6. opcache.enable_cli=1
  7. ; Zend Optimizer + 共享内存的大小, 总共能够存储多少预编译的 PHP 代码(单位:MB)
  8. ;根据内存来定
  9. opcache.memory_consumption=256
  10. ; Zend Optimizer + 暂存池中字符串的占内存总量.(单位:MB)
  11. ; 推荐 8
  12. opcache.interned_strings_buffer=4
  13. ; 最大缓存的文件数目 200 100000 之间
  14. ; 推荐 4000~8000
  15. opcache.max_accelerated_files=8000
  16. ; 内存“浪费”达到此值对应的百分比,就会发起一个重启调度.
  17. opcache.max_wasted_percentage=5
  18. ; 开启这条指令, Zend Optimizer + 会自动将当前工作目录的名字追加到脚本键上,
  19. ; 以此消除同名文件间的键值命名冲突.关闭这条指令会提升性能,
  20. ; 但是会对已存在的应用造成破坏.
  21. opcache.use_cwd=0
  22. ; 开启文件时间戳验证
  23. opcache.validate_timestamps=1
  24. ; 2s检查一次文件更新 注意:0是一直检查不是关闭
  25. ; 推荐 60
  26. opcache.revalidate_freq=0
  27. ; 允许或禁止在 include_path 中进行文件搜索的优化
  28. ;opcache.revalidate_path=0
  29. ; 是否保存文件/函数的注释 如果apigenDoctrine ZF2 PHPUnit需要文件注释
  30. ; 推荐 0
  31. opcache.save_comments=1
  32. ; 是否加载文件/函数的注释
  33. ;opcache.load_comments=1
  34. ; 打开快速关闭, 打开这个在PHP Request Shutdown的时候会收内存的速度会提高
  35. ; 推荐 1
  36. opcache.fast_shutdown=1
  37. ;允许覆盖文件存在(file_exists等)的优化特性。
  38. ;opcache.enable_file_override=0
  39. ; 定义启动多少个优化过程
  40. ;opcache.optimization_level=0xffffffff
  41. ; 启用此Hack可以暂时性的解决”cant redeclare class”错误.
  42. ;opcache.inherited_hack=1
  43. ; 启用此Hack可以暂时性的解决”cant redeclare class”错误.
  44. ;opcache.dups_fix=0
  45. ; 设置不缓存的黑名单
  46. ; 不缓存指定目录下cache_开头的PHP文件. /png/www/example.com/public_html/cache/cache_
  47. ;opcache.blacklist_filename=
  48. ; 通过文件大小屏除大文件的缓存.默认情况下所有的文件都会被缓存.
  49. ;opcache.max_file_size=0
  50. ; N 次请求检查一次缓存校验.默认值0表示检查被禁用了.
  51. ; 由于计算校验值有损性能,这个指令应当紧紧在开发调试的时候开启.
  52. ;opcache.consistency_checks=0
  53. ; 从缓存不被访问后,等待多久后(单位为秒)调度重启
  54. ;opcache.force_restart_timeout=180
  55. ; 错误日志文件名.留空表示使用标准错误输出(stderr).
  56. ;opcache.error_log=/tmp/ckl.log
  57. ; 将错误信息写入到服务器(Apache等)日志
  58. ;opcache.log_verbosity_level=1
  59. ; 内存共享的首选后台.留空则是让系统选择.
  60. ;opcache.preferred_memory_model=
  61. ; 防止共享内存在脚本执行期间被意外写入, 仅用于内部调试.
  62. ;opcache.protect_memory=0

参考下大神的配置

  1. opcache.enable=1
  2. opcache.memory_consumption=256
  3. opcache.interned_strings_buffer=4
  4. opcache.max_accelerated_files=8000
  5. opcache.max_wasted_percentage=5
  6. opcache.use_cwd=1
  7. opcache.validate_timestamps=1
  8. opcache.revalidate_freq=0
  9. opcache.revalidate_path=0
  10. opcache.save_comments=0
  11. opcache.load_comments=0
  12. opcache.force_restart_timeout=3600

性状、生成器、闭包、OPcache【Modern PHP】的更多相关文章

  1. Python四大神兽(迭代器&生成器&闭包&装饰器)

    一.迭代器 迭代是Python最强大的功能之一,是访问集合元素的一种方式.. 迭代器是一个可以记住遍历的位置的对象. 迭代器对象从集合的第一个元素开始访问,直到所有的元素被访问完结束.迭代器只能往前不 ...

  2. 05. Go 语言函数

    Go 语言函数 函数是组织好的.可重复使用的.用来实现单一或相关联功能的代码段,其可以提高应用的模块性和代码的重复利用率. Go 语言支持普通函数.匿名函数和闭包,从设计上对函数进行了优化和改进,让函 ...

  3. python函数总结,你值得拥有

    目录 函数总结 函数定义与结构 函数名的使用 函数的参数 名称空间与作用域 名称空间 作用域 函数嵌套 内置函数(globals( ),locals( )) global+nonlocal 可迭代对象 ...

  4. php中trait(性状)与generator(生成器)

    PHP中trait(性状)与generator(生成器) 一.trait (性状) 最近在看Josh Lockhat的<Modern PHP>,这本书很薄.但是其中给出了一个很重要的学习方 ...

  5. Day4 闭包、装饰器decorator、迭代器与生成器、面向过程编程、三元表达式、列表解析与生成器表达式、序列化与反序列化

    一.装饰器 一.装饰器的知识储备 1.可变长参数  :*args和**kwargs def index(name,age): print(name,age) def wrapper(*args,**k ...

  6. Python编程四大神兽:迭代器、生成器、闭包和装饰器

    生成器 生成器是生成一个值的特殊函数,它具有这样一个特点:第一次执行该函数时,先从头按顺序执行,在碰到yield关键字时该函数会暂停执行该函数后续的代码,并且返回一个值:在下一次调用该函数执行时,程序 ...

  7. 小白Python路上第一个难点,也是一个比较重要的点(闭包,迭代器,生成器)

    一.闭包 闭包就是在内层函数中引用外层函数的变量 作用:1.保护变量不受侵害          2.让一个变量永驻内存 二.迭代器 Iterator:迭代器,包含_iter_()和_next_()函数 ...

  8. Python核心编程的四大神兽:迭代器、生成器、闭包以及装饰器

      生成器 生成器是生成一个值的特殊函数,它具有这样的特点:第一次执行该函数时,先从头按顺序执行,在碰到yield关键字时该函数会暂停执行该函数后续的代码,并且返回一个值:在下一次调用该函数执行时,程 ...

  9. Python之路【第五篇】: 函数、闭包、装饰器、迭代器、生成器

    目录 函数补充进阶 函数对象 函数的嵌套 名称空间与作用域 闭包函数 函数之装饰器 函数之可迭代对象 函数之迭代器 函数之生成器 面向过程的程序设计思想 一.函数进阶之函数对象 1. 函数对象 秉承着 ...

随机推荐

  1. OpenLdap与BerkeleyDB安装过程

    前段时间在看LDAP方面的东西,最近重装了Ubuntu之后开始在自己的机器上装一个OpenLDAP. 装的过程中还遇到不少问题,不过通过Google同学的帮忙,全部得到解决.下面装安装过程记录如下: ...

  2. ognl,jstl,struts2标签中符号#,$,%的用法

    STRUTS2标签操作Map <s:iterator value="sundayMap">           <td colspan="7" ...

  3. [PY3]——内置数据结构(2)——元组及其常用操作

    定义和初始化 #tuple() 使用工厂函数tuple定义一个空元组 #() 使用圆括号定义一个空元组 #(1,2,3) 使用圆括号定义有初始值的元组 #tuple(可迭代对象) 把可迭代对象转换为一 ...

  4. Spring MVC 数据绑定流程分析

    1.    数据绑定流程原理★ ①   Spring MVC 主框架将 ServletRequest  对象及目标方法的入参实例传递给 WebDataBinderFactory 实例,以创建 Data ...

  5. [转]MongoDB 概念解析

    本文转自:http://www.runoob.com/mongodb/mongodb-databases-documents-collections.html 不管我们学习什么数据库都应该学习其中的基 ...

  6. ASP.NET之Jquery入门级别

    1.Jquery的简单介绍 1)Jquery由美国人John Resig创建.是继prototype之后又一个优秀的JavaScript框架. 2)JQuery能做什么?JQuery能做的普通的Dom ...

  7. 十七、ThreadPoolExecutor线程池

    一.简介 executor接口 executor接口在JDK的java.util.concurrent包下,它只有一个抽象方法: void execute(Runnable command); 这意味 ...

  8. 二、hdfs单节点安装

    一.准备环境 在配置hdfs之前,我们需要先安装好hadoop的配置,本文主要讲述hdfs单节点的安装配置. hadoop的单节点安装配置请参考:https://www.cnblogs.com/lay ...

  9. 用JavaScript将数字转换为大写金额(收藏)

    (非原创, 来自网络,仅作收藏) var digitUppercase = function(n) { var fraction = ['角', '分']; var digit = [ '零', '壹 ...

  10. 撩课-Java每天10道面试题第1天

    1.简述JDK.JRE.JVM? 一.JDK JDK(Java Development Kit) 是整个JAVA的核心, 包括了Java运行环境(Java Runtime Envirnment), 一 ...