转载自 http://developer.51cto.com/art/200903/115995.htm
http://developer.51cto.com/art/200903/115995_1.htm
一、mysql驱动mysqlnd
一直以来,php都是通过mysql客户端连接mysql,而现在mysql官方已经推出php版的mysql客户端,而这个mysqlnd有效降低内存的使用以及提高性能。具体可以看:
http://dev.mysql.com/downloads/connector/php-mysqlnd/
http://forge.mysql.com/wiki/PHP_MYSQLND
从图中可以看出,使用mysqlnd少了从mysql驱动中复制数据到php扩展这一步。mysqlnd使用copy-on-write,也就是写时复制,读引用。
mysqlnd已经内置在php5.3的源码中,编译的时候使用--with-mysql=mysqlnd、--with-mysqli=mysqlnd 和 --with-pdo-mysql=mysqlnd 安装mysqlnd驱动。
mysqlnd的优点
编译php更方便了,不需要libmysql,已经内置在源码中
- 编译php更方便了,不需要libmysql,已经内置在源码中
- 使用php许可,避免版权问题
- 使用php的内存管理,支持php内存限制(memory_limit)
- 所有数据在内存只有一份,之前的libmysql有两份,参考上图
- 提供性能统计功能,帮助分析瓶颈
- mysqli支持长连接(persistent connections)
- 性能绝对比libmysql要快
- 在驱动层增加缓存机制
看了这么多特点,有点矛盾,作为数据库抽象层的PDO能把不同后端的特点发挥出来吗?如果使用mysql作为数据库的话mysqli是不是更好的选择?我总觉得mysqli只是个过度产品,PDO才是未来的。
二、性能提高
- md5()大概提高了10%-15%的性能
- Better stack implementation in the engine,没明白
- 常量保存在ROM里(Constants moved to read-only memory),这里没明白意思
- 改进异常处理,操作码(opcode)更简洁
- 解决了include(require)_once重复打开的问题,之前once我都是用静态变量实现的,终于解决这个问题了
- 用gcc4编译的二进制文件将更小
- 整体性能提高了5%-15%
很多人觉得web的瓶颈在db,所以app应用的性能毫不在乎,我认为主要是因为app扩展比db扩展要容易得多,所以才会产生db瓶颈,但这并不意味着可以不顾虑app的性能,毕竟最终还是要在app里解决各种问题的,作为一名程序员编写高质量的代码是最基本的要求。程序的内存使用量少点和执行速度快点,在高并发下是有很效果的,有些时候你改动下实现方法,能提高几十倍也很正常,当然如果付出要很多收获很少的时候就不要太执着了,我觉得一定要有编写高质量代码的意识。
三、?:操作符
其实就是js里的||,返回的结果不是逻辑类型,而是返回本来变量的值,例如 false ?: 123返回123,而不是true。语法就是有点怪怪!
四、名字空间(namespace)
这是个很好的功能,没加入之前都是用前缀来解决命名污染的,方法有点山寨,呵呵。
五、延迟静态绑定(Late Static Binding)
我估计php的静态是在预编译时就固定好的,所以在继承的时候,父类里的self指的是父类,而不是子类。而php5.3加入了新的语法static,可以在运行时候捕捉当前类。
比较典型的例子就是单件模式了:
class ParentClass { static private $_instance; private function __construct() { } static public function getInstance() { if (!isset(self::$_instance)) { self::$_instance = new self(); } return self::$_instance; } } |
如果继承这个父类,那子类的单件必须再重写覆盖父类的getInstance。5.3支持用static来延迟绑定,但是很遗憾,即使有static关键字却不能使用new static来实例化,不过有变通的方法
class ParentClass { static private $_instance; private function __construct() { } static public function getInstance() { if (!isset(self::$_instance)) { $class = static::getClass();// 使用static延迟绑定 self::$_instance = new $class(); } return self::$_instance; } static public function getClass() { return __CLASS__; } } |
这个新特性当前可能会有问题存在,暂时不建议使用。
六、新的魔法函数__callStatic
其实就是__call的静态版,调用的静态方法不存在会调用这个魔法函数,但是低效
七、通过变量调用静态(Variable Static Calls)
以前可以通过变量调用对象的方法,例如$instance->$method();,而静态方法不支持,5.3后就支持了可以通过$someClass::$method()调用了,但是低效。
八、日期函数date_create_from_format
把字符串转换成时间戳,如果使用strtotime的话,日期格式是php解析的,而5.3的date_create_from_format可以设置字符串的日期格式,
$date = strtotime("08-01-07 00:00:00");//php 认为格式 是年-月-日 var_dump(date("Y-m-d", $date)); // string(10) "2008-01-07" $date = date_create_from_format("m-d-y", "08-01-07");//告诉php格式是 月-日-年 var_dump($date->format('Y-m-d')); // string(10) "2007-08-01" |
九、匿名函数(Lambda functions)和闭包(closures)
js有个很爽的地方就是支持函数式编程,php很不爽的地方就是函数太过正规了,不过5.3后php更自由了。
定义匿名函数的语法
$lambda = function () { echo "Hello World!\n"; }; |
function replace_spaces ($text) { $replacement = function ($matches) { return str_replace ($matches[1], ' ', ' ').' '; }; return preg_replace_callback ('/( +) /', $replacement, $text); } |
function replace_spaces ($text) { return preg_replace_callback ('/( +) /', function ($matches) { return str_replace ($matches[1], ' ', ' ').' '; }, $text); } |
产生闭包的语法
function (normal parameters) use ($var1, $var2, &$refvar) {} |
引入了use语法,并且支持引用变量
十、新魔法常量__DIR__
以前只有__FILE__,为了得到当前文件所在的路径,都是用dirname(__FILE__)来得到的,现在可以用__DIR__代替了
十一、NOWDOC
php定义字符串有一种格式叫定界符.
$foo = < |
这种方式称为HEREDOC,php会解析里面的变量,而有时候我们不需要解析变量,5.3加入了NOWDOC,其实就是定界符的单引号版
$bar = <<<'TWO' this is $fubar TWO; |
这样,php只把它当作字符串,变量不会解析
十二、GC
因为php本身工作机制的原因,所以php的GC机制只要高效就够了,允许少量内存泄漏,毕竟程序执行完毕就会释放所有申请的内存,根本不会有内存泄漏的问题,但这只是对于短暂运行的程序而言,如果用php编写持久执行就要考虑内存泄漏问题了。
php的GC机制采用引用计数机制,引用计数是很简单高效的GC机制,不过缺点也很明显,不能完全回收所有已无效的变量,例如变量相互引用了,就无法回收了,5.3里加入的GC函数,其实就是起到增强GC机制的作用。
gc_enable(); // 激活GC,增强GC机制,回收循环引用的无效变量 var_dump(gc_collect_cycles()); // 强制回收已无效的变量 gc_disable(); // 禁用GC |