PHP(四)——性能优化

之前基于PHP开发的过程中,一直没有涉及到PHP性能优化的问题,但是一般来说PHP性能问题占整个项目性能问题一般占30%-50%部分,所以说,这部分内容是非常重要的。下面是最近自己PHP性能优化学习的资料整理。

引言[1]

  • PHP存在性能问题的情景

    • PHP语法使用不恰当
    • 使用了PHP语言他不擅长做的事情
    • 用PHP语言连接的服务不给力
    • PHP自身的短板
    • 未知的问题
  • PHP性能问题解决方向

    • PHP语言级的性能优化:日常语法方法的优化。特点:简单高效很快见到效果。
    • PHP周边问题的性能优化 :webserver,mysql。
    • PHP语言自身的分析和优化 :PHP底层C语言逻辑的优化。

从1~5的顺序,按照操作简单,见效快的指标进行的解决方案的排序。

  • Apache压力测试软件

    • Apache Benchmark,简称ab,是由Apache提供的压力测试软件,安装apache服务器时会自带压测软件。

    • 使用./ab -n100 -c10 http://www.baidu.com
      其中-n请求数,-c并发数,url目标。

    • ab 返回结果的参数

      • Requests per second 每秒请求数 (优化目标 每秒的请求数尽可能多)
      • Time per request 响应一个请求耗时 (优化目标 响应一个请求尽可能少)

PHP语言级性能优化

  • 基本原则:少写php代码,多使用PHP内置的变量、常量、函数。

  • 性能问题:自写代码冗余较多,可读性不佳,并且性能低。php代码写的越长长执行效果就会越差,多用php自身的函数等

    因为PHP代码需要编译为C语言,C语言又会编译成汇编语言(机器语言),这里每一个过程都会请求一遍,开销很大。尤其是访问量大的时候,每次都会编译一遍。所以要尽量减少代码。

  • PHP代码如何再linux解析的流程

    • 逐行扫描。*.php 通过zend引擎逐行扫描分析(Scanner)
    • 转码。保存成zend引擎自己能识别的语法(Exprs)
    • 解析为Opcodes。这些zend引擎能识别的语法,再解析(Parser)成Opcodes(最终要拿去执行的机器代码)。
    • 输出。执行Opcodes,然后输出。

      缓存服务都是缓存的Opcodes,不需要扫描和解析,

      PHP_process

PHP语言级性能优化建议

  • PHP内置函数的性能优劣
    • 尽量使用更快内置函数,不同函数依然存在快慢差异。
    • 尽量少用魔法函数,魔法函数性能低。为了给程序员省事,php语言为你做了很多linux time函数 可以直接测试程序的耗时情况 魔法函数举例:__get();可以不用尽量不用,如果必须要用的时候再用。
    • 不使用@错误抑制符,改用try throw
  • 合理使用内存
    • 情况描述:php有内存回收机制保底,但也要小心使用内存
    • 建议:利用unset()及时释放不使用的内存(注:unset出现注销不掉的情况,自己查资料)
  • 尽量少的使用正则表达式
    • 情况描述:正则表达式性能低,因为正则表达式回溯开销较大
    • 好的建议:利用字符串处理函数,实现相同的逻辑
  • 避免循环内做运算
    • 情况描述:循环内的计算式会被重复计算
    • 例 for($i=0;$i<strlen($str);$i++) 每一次for循环都会进行计算strlen
  • 减少计算密集型业务
    • 情况描述:PHP不适合密集型运算场景
    • 原因:
      • 比如不适合大批量日志分析,或者大批量数据处理。
      • php语言特性决定了PHP不适合做大数据运算
      • php所有处理都需要转换成C语言,与C相比,C更好。
      • php还有环境问题,还有语言特性。额外开销比C大很多。变量寄存等。
    • PHP适合处理场景:适合衔接webserver与后端服务、UI呈现。(就是接口,简单数据处理,和套页面)
  • 务必使用带引号字符串做键值:php会将没有引号的键值检查一遍是不是常量,产生查询常量的开销

PHP周边问题的性能优化

  • PHP周边

    • linux环境
    • 硬盘,文件存储,php读写
    • 数据库
    • 缓存:缓存是基于内存的
    • 网络
  • PHP部署环境优化

    单台服务器常用apache+phpnginx+php-fpm方式部署,据说现在用nginx+php-fpm部署方式性能比apache+php性能好,可考虑一试。另外如nginx+swoole等,也是可选项。

    集群是在此基础上,使用nginx/lvs/云上lbs等反向代理作为负载均衡前端。PHP集群部署在可靠性的基础上,PHP集群处理性能比单台服务器有N倍提高(但作为服务的整体性能并不一定有N倍提升)。所以简单地可以认为,通过集群扩展服务器,可以使PHP服务性能得到提升。

性能优化推荐

  • 减少文件类操作。常见PHP场景的开销次序:读写内存 < 读写数据库 < 读写磁盘 < 读写网络数据
  • 减少php发起网络请求,优化网络请求

    • 网络请求的坑:1对方接口的不确定因素 2 网络稳定性
    • 优化网络请求方法

      • 设置超时时间(建议值)

        • 连接超时 200ms 这是上限,最多也不能超过这个时间
        • 读超时 800ms 这个看具体情况
        • 写超时 500ms 建议不要超过500ms
      • 将串行请求并行化

        • 使用curl_multi_*() 返回时间是看用时最长的那个请求
        • 使用swoole扩展,通过C来进行并行化。(推荐使用)
  • 压缩PHP接口输出:如果用php做接口可通过使用Gzip压缩实现更高效的输出。压缩输出的利弊:
    • 利:利于数据输出,client能更快获取数据
    • 弊:额外的CPU开销。如果请求大,可能会有问题

      gzip如果数据量小于几十K的时候效果并不理想。如果大于100k,压缩就有效果。

  • 缓存重复计算内容:固定重复请求的数据做缓存。
  • 重叠时间窗口思想:串行变并行。如果后一个请求不强依赖于前一个返回值。就可以变成并行,降低总体时间消耗

PHP语言自身的分析和优化

  • PHP扩展使用[2],PHP扩展除了使用方便,还是提升性能的亲密伙伴。主要应用有三点:

    • 开启opcode的缓存,来避免重复的编译。可以使用APC,eAccelerator,XCache等PHP扩展,我们使用xcache。这种只要安装即可。

    • 使用扩展提供的方法(或PHP标准库的方法)。通过PHP扩展代替原PHP代码中高频的逻辑,扩展实现的效率比PHP代码中的高。但实际上满足我们项目的扩展方法有限,很多基础方法需要时一步封装,除非有能力自己开发扩展。可考虑使用扩展实现的PHP框架,如phalcon、yaf。

    • 本地缓存,也常用扩展来支持,比如xcache。本地可使用缓存扩展,缓存一些配置数据、元数据或主数据,不用每次都从数据库或文件中读取。

另外,PHP版本上,可以考虑升到PHP7,PHP7在性能上有很大的提升。

  • Runtime优化:HHVM[3](phpng也许更优于HHVM)

具体的PHP语言级的优化建议[4]

  1. 用单引号替代双引号引用字符串,这样做会更快一些。因为PHP会在双引号包围的字符串中搜寻变量,单引号则不会。

  2. 如果能将类的方法定义成static,就尽量定义成static,它的速度会提升将近4倍。

  3. $row[‘id’] 的速度是$row[id]的7倍。

  4. echo 比 print 快,并且使用echo的多重参数(译注:指用‘,’号而不是‘.’)代替字符串连接,比如echo $str1,$str2。区别用‘.’,先拼接,在整个输出;用‘,’,是挨个把三个变量输出。

  5. 在执行for循环之前先确定最大循环数,不要每循环一次都计算最大值。

    1
    2
    // 不合理,每次循环都要计算
    for($i=0;$i<strlen($str);$i++)
  6. foreach效率更高,尽量用foreach代替while和for循环,如果考虑到foreach($array as $var)每次拷贝的消耗,可以使用foreach($array as &$var)这样的引用。

  7. 尽量避免使用魔术变量,如__get__set__autoload

  8. require_once()代价昂贵。require_onceinclude_once需要判重,因此效率上要低,但是5.2版本后效率问题已经基本解决。

  9. include文件时尽量使用绝对路径,因为它避免了PHP去include_path里查找文件的速度,解析操作系统路径所需的时间会更少。尽量少用iniset()来设置include_path

  10. 返回脚本开始执行(即服务器端收到客户端请求)的时刻,使用$_SERVER['REQUEST_TIME']要好于time()。因为$_SERVER['REQUEST_TIME']保存了发起该请求时刻的时间戳,而time()则返回当前时刻的Unix时间戳。

  11. 函数代替正则表达式完成相同功能,字符串操作比正则替换要快。如strtokstrstrstrposstr_replacesubstrexplodeimplode等等。注意不同的函数快慢也不同,str_replace函数比preg_replace函数快,但strtr函数的效率是str_replace函数的四倍。

  12. 使用选择分支语句(译注:即switch case)好于使用多个if,else if语句。因为php中switch支持数值和字符串变量,比C的switch要好用,建议使用。
  1. 数据库连接当使用完毕时应关掉,不要用长连接。建议在连接之前,最好设置一下相应的超时机制,例如链接超时、读写超时、等待超时等。

  2. 错误消息代价昂贵。所以说在代码测试完成,上线之前删除错误信息报告代码。

  3. 在方法中递增局部变量,速度是最快的。几乎与在函数中调用局部变量的速度相当。递增一个全局变量要比递增一个局部变量慢2倍。递增一个对象属性(如:$this->prop++)要比递增一个局部变量慢3倍。递增一个未预定义的局部变量要比递增一个预定义的局部变量慢9至10倍。仅定义一个局部变量而没在函数中调用它,同样会减慢速度(其程度相当于递增一个局部变量)。

  4. 派生类中的方法运行起来要快于在基类中定义的同样的方法。

  5. Apache解析一个PHP脚本的时间要比解析一个静态HTML页面慢2至10倍。尽量多用静态HTML页面,少用脚本。

  6. 除非脚本可以缓存,否则每次调用时都会重新编译一次。引入一套PHP缓存机制通常可以提升25%至100%的性能,以免除编译开销。

  7. 尽量可使用memcached等做缓存。memcached是一款高性能的内存对象缓存系统,可用来加速动态Web应用程序,减轻数据库负载。对运算码(OPcode)的缓存很有用,使得脚本不必为每个请求做重新编译。

  8. 当执行变量$i的递增或递减时,$i++会比++$i慢一些。这种差异是PHP特有的,并不适用于其他语言。++$i更快是因为它只需要3条指令(opcodes),$i++则需要4条指令。后置递增实际上会产生一个临时变量,这个临时变量随后被递增。而前置递增直接在原值上递增。这是最优化处理的一种,正如Zend的PHP优化器所作的那样。

  9. 面向对象(OOP)是非必要的,因为面向对象往往开销很大,每个方法和对象调用都会消耗很多内存。并非要用类实现所有的数据结构,数组也很有用。

  10. 不要把方法细分得过多,仔细想想你真正打算重用的是哪些代码?不要过分迷恋各种设计模式,如上一条描述,过分的封装会带来性能的下降。需要考虑两者的权衡。Php有自己的特点,切不可东施效颦,过分效仿java的模式。

  11. 分解成方法要适当,行数少使用频率高的方法尽量用直接写代码,可以减少函数堆栈开销;且方法嵌套不宜过深,否则大大影响PHP的运行效率。

  12. 尽量采用大量的PHP内置函数,除去空函数调用的影响,内置函数和同样功能的C函数性能基本差不多。

  13. 如果在代码中存在大量耗时的函数,你可以考虑用C扩展的方式实现它们。

  14. 打开apache的mod_deflate模块,可以提高网页的浏览速度。mod_zip可作为Apache模块,用来即时压缩你的数据,并可让数据传输量降低80%。

  15. 在可以用file_get_contents替代file、fopen、feof、fgets等系列方法的情况下,尽量用file_get_contents,因为他的效率高得多!但是要注意file_get_contents在打开一个URL文件时候的PHP版本问题。这个要记住,尽量使用file_get_contentsfile_put_contents,不需要自己判断文件句柄打开是否成功。

  16. 尽量的少进行文件操作,虽然PHP的文件操作效率也不低的;

  17. 优化Select SQL语句,在可能的情况下尽量少的进行Insert、Update操作(在update上,我被恶批过);

  18. 循环内部不要声明变量,尤其是大变量:对象(这好像不只是PHP里面要注意的问题吧?)。这个必须的,变量过多或者过大时,每次重分配的开销就无法忽略。

  19. 多维数组尽量不要循环嵌套赋值;

  20. 对global变量,应该用完就unset()释放掉;

  21. 当操作字符串并需要检验其长度是否满足某种要求时,使用isset()替代strlen()函数。isset()作为一种语言结构,它的执行不需要函数查找和字母小写化。此函数执行起来相当快,只返回在zval结构(C的内置数据结构,用于存储PHP变量)中存储的已知字符串长度。但是,由于strlen()是函数更慢,因为函数调用会经过诸多步骤,如字母小写化(指函数名小写化,PHP不区分函数名大小写)、哈希查找,会跟随被调用的函数一起执行。在某些情况下,你可以使用isset() 技巧加速执行你的代码。   

    1
    2
    3
    4
    // (举例如下)
    if (strlen($foo) < 5) { echo “Foo is too short”$$ }
    // (与下面的技巧做比较)
    if (!isset($foo{5})) { echo “Foo is too short”$$ }
  22. 评估检验(profile)你的代码。检验器会告诉你,代码的哪些部分消耗了多少时间。Xdebug调试器包含了检验程序,评估检验总体上可以显示出代码的瓶颈。

  23. 函数相关信息保存在一个大的hash_table中,每次调用时通过函数名在hash表中查找,因此函数名长度对性能也有一定影响。

  24. 函数不宜嵌套过深,递归使用要谨慎。

  25. 如不是特殊需要,参数传递都建议使用传值而不是传引用。当然,如果参数是很大的数组且需要修改时可以考虑引用传递。

  26. 使用NoSQL、Memchached或者Redis缓存。这些是高性能的分布式内存对象缓存系统,能提高动态网络应用程序性能,减轻数据库的负担。这对运算码 (OPcode)的缓存也很有用,使得脚本不必为每个请求重新编译。

参考

[1] 慕课网 – 性能优化之PHP优化总结笔记,视频地址

[2] PHP 性能优化

[3] 关于 PHP 性能优化

[4] 一些PHP性能的优化

[5] 48条高效率的PHP优化写法

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/242267.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C#常用的算法

C#常用的算法 遍试算法 迭代算法 递归 递归算法的基本思想&#xff1a;分而治之

C#流和文件

C#流和文件 文件及文件夹管理 WinForm中的文件对话框 文件内容的读写 注册表操作

C#中程序错误的种类和程序的调试

C#中程序错误的种类 语法错误 运行错误 逻辑错误 程序调试的基本手段 然后运行 调试程序例子 例子2&#xff1a;读文件 没有找到文件 null引用就是没有进行初始化 索引超出数组界限

C#中window窗体和控件

C#中window窗体和控件 布局与事件–界面的基本问题 布局 就是拉动窗体的时候&#xff0c;按钮也在跟着变动。 事件 //简单的计算器 using System; using System.Drawing; using System.Collections; using System.ComponentModel; using System.Windows.Forms; namespa…

C#常用的控件

C#常用的控件 窗体 菜单、工具栏、对话框 用户控件及控件的继承

C#绘图及图像

C#绘图及图像 绘图的基本方法 绘图的应用

C#实现定时器

C#实现定时器 方法一 布局 代码 using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Text; using System.Threading.Tasks; using System.Windows.Forms;namespace…

mark

这里写自定义目录标题欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants创建一个自定义列表如何创建一个注…

C语言的进阶-指针的应用

指针的应用 #include <stdio.h> void swap(int *p,int *q); int main() {int a 5;int b 8;swap(&a,&b);printf("a%d,b%d\n",a,b);return 0; }void swap(int *p,int *q) {int t *p;*p *q;*q t; }函数只能返回一个值&#xff01; 可以通过参数返回多…

C语言进阶-指针与数组

C语言进阶-指针与数组 q可以改变指向的内容&#xff0c;但不能改变指向的地址 p可以改变指向地址&#xff0c;但不能改变指向内容

发布文章测试

这里写自定义目录标题欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants创建一个自定义列表如何创建一个注…

C语言进阶-指针运算

C语言进阶-指针运算 #include <stdio.h>int main() {char a[] {0,1,2,3,4,5,6,7,8,9};char *p a;printf("p%d\n",p);printf("p1%d\n",p1);int b[] {0,1,2,3,4,5,6,7,8,9};int *q b;printf("q%d\n",q);printf("q1%d\n",q1);r…

C语言进阶-动态内存分配

C语言进阶-动态内存分配 #include <stdio.h> #include<stdlib.h>int main() {int number;int *a;int i;printf("请输入数组的数量");scanf("%d",&number);//int a[number];a(int*)malloc(number*sizeof(int));//开辟数组的大小for(i 0;i&…

测测

编辑 1.多喝水 2.清黑头&#xff0c;一周做一次角质&#xff0c;一个月做一次小气泡 3.防晒 无论春夏秋冬 4.少吃辣&#xff0c;多喝柠檬水 5.早上护肤步骤&#xff1a;洁面-爽肤水-眼霜-精华-面霜-防晒 6.晚上护肤步骤&#xff1a;卸妆-洁面-水-眼霜-精华-面霜 7.一周三…

QT输出出现乱码的解决

QT输出出现乱码的解决

测试图片外链MD

11111 11111111111111111111111111111 555555555

测试阿里云HTML

一直在说事件&#xff0c;那么事件到底是指什么&#xff1f;这里所说的事件是指手指按下(down)、移动(move)、抬起(up)此为一个事件集合或者说是事件序列&#xff0c;从手指接触屏幕开始到手指离开屏幕结束。所以本篇所说的事件序列或者事件集合是指从手指刚接触屏幕到离开屏幕…

测试阿里云MD

准备工作已经完成&#xff0c;闲言少叙书归正传吧。 和拦截处理机制详解一样&#xff0c;为了系统的研究android对事件的处理&#xff0c;我也写了一个小demo对不同的情况进行测试并结合源码分析&#xff08;多说一句&#xff0c;其实看源码确实很枯燥&#xff0c;有时候因为水…