您现在的位置是:首页 > 文章内容文章内容

PHP性能提升与优化

lipeng2019-08-27 14:34:09PHP1人已围观



1、用单引号代替双引号来包含字符串,这样做会更快一些。因为PHP会在双引号包围的字符串中搜寻变量,单引号则不会,注意:只有echo能这么做,它是一种可以把多个字符串当作参数的"函数"(译注:PHP手册中说echo是语言结构,不是真正的函数,故把函数加上了双引号)。

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

3、$row['id'] 的速度是$row[id]的7倍。

4、echo 比 print 快,并且使用echo输出时用逗号而不是句点代替字符串连接效率更高,比如 echo $str1,$str2。

5、在执行for循环之前确定最大循环数,不要把 count/strlen/sizeof 等每次都要重复做的但结果都一样的事情放到 for 循环的条件语句中,另外最好运用foreach代替for循环。

6、及时注销那些不用的变量,尤其是大数组,对象之类的,以便释放内存。

7、require_once()代价昂贵,据测试数据来看,使用require_once比require慢3-4倍,具体的解决办法可先检查是否存在引用然后决定是都需要require。

8、include和require文件时尽量不要使用相对路径,因为使用相对路径的时候它会首先查找指定的php包含路径,然后查找当前目录,因此会检查过多路径,所以最佳选择是使用绝对路径。

9、如果你想知道脚本开始执行(即服务器端收到客户端请求)的时间,使用$_SERVER['REQUEST_TIME']要好于time()。至于$_SERVER['REQUEST_TIME']的作用,文档解释是该变量保存的是页面请求开始时的时间戳。从 PHP 5.1.0 起有效。和time函数效果一样。

10、能够使用函数代替正则表达式的地方尽量使用函数来完成。

11、str_replace函数比preg_replace函数快,但strtr函数的效率是str_replace函数的四倍。strtr() 函数的作用是转换字符串中特定的字符。

12、不要做无谓的替换,即使没有替换操作,使用 str_replace 也会为其参数分配内存。很慢!解决办法:用 strpos 先查找相关信息看是否需要替换,如果需要,再替换。实际效率对比为:如果需要替换:效率几乎相等,差别在 0.1% 左右。如果不需要替换:用 strpos 速度将提升 200%。

13、如果一个字符串替换函数,可接受数组或字符作为参数,并且参数长度不太长,那么可以考虑额外写一段替换代码,使得每次传递参数是一个字符,而不是只写一行代码接受数组作为查询和替换的参数。

14、使用选择分支语句(即switch case)好于使用多个if,else if语句。

15、不要滥用 @ 操作符。虽然 @ 看上去很简单,但是实际上后台有很多操作。用 @ 比起不用 @,效率差距:3 倍。特别不要在循环中使用 @ 。

16、打开apache的mod_deflate模块,可以提高网页的浏览速度。mod_deflate 模块提供了DEFLATE 输出过滤器,允许服务器在将输出内容发送到客户端以前进行压缩,以节约带宽。具体如何设置请参考相关文档。

17、在方法中递增局部变量,速度是最快的。几乎与在函数中调用局部变量的速度相当。而递增一个全局变量要比递增一个局部变量慢2倍。

18、在方法中递增一个对象属性(如:$this- >num++)要比递增一个局部变量(如:$num)慢3倍。

19、递增一个未预定义的局部变量要比递增一个预定义的局部变量慢9至10倍。

20、仅定义一个局部变量而没在函数中调用它,同样会减慢速度(其程度相当于递增一个局部变量)。PHP大概会检查看是否存在全局变量。

21、方法调用看来与类中定义的方法的数量无关,因为我(在测试方法之前和之后都)添加了10个方法,但性能上没有变化。

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

23、调用带有一个参数的空函数,其花费的时间相当于执行7至8次的局部变量递增操作。类似的方法调用所花费的时间接近于15次的局部变量递增操作。

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

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

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

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

<?php
  if (strlen($str) < 6){ 
    echo 'str不满6个字符'; 
  } ?>


(与下面的技巧做比较)

<?php
  if (!isset ($str{6})){
    echo 'str不满6个字符';
  } 
?>



调用isset()比strlen()快,因为isset()作为一种语言结构,意味着它的执行不需要函数查找和字母小写化。也就是说,实际上在检验字符串长度的顶层代码中你没有花太多开销。

28、当执行变量$i的递增或递减时,$i++会比++$i慢一些。这种差异是PHP特有的,并不适用于其他语言,所以请不要修改你的C或Java代码并指望它们能立即变快,没用的。++$i更快是因为它只需要3条指令(opcodes),$i++则需要4条指令。后置递增实际上会产生一个临时变量,这个临时变量随后被递增。而前置递增直接在原值上递增。这是最优化处理的一种,正如Zend的PHP优化器所作的那样。牢记这个优化处理不失为一个好主意,因为并不是所有的指令优化器都会做同样的优化处理,并且存在大量没有装配指令优化器的互联网服务提供商和服务器。

29、并不是事必面向对象(OOP),面向对象往往开销很大,每个方法和对象调用都会消耗很多内存。

30、并非要用类实现所有的数据结构,数组也很有用。

31、不要把方法细分得过多,仔细想想你真正打算重用的是哪些代码。

32、能使用PHP内置函数的地方尽量使用PHP内置函数。

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

34、评估检验(profile)你的代码。检验器会告诉你,代码的哪些部分消耗了多少时间。Xdebug调试器包含了检验程序,评估检验总体上可以显示出代码的瓶颈。

35、mod_zip可作为Apache模块,用来即时压缩你的数据,并可让数据传输量降低80%。

36、在可以用file_get_contents替代file、fopen、feof、fgets等系列方法的情况下,尽量用file_get_contents,因为他的效率高得多!但是要注意file_get_contents在打开一个URL文件时候的PHP版本问题;

37、尽量的少进行文件操作,虽然PHP的文件操作效率也不低的。

38、优化Select SQL语句,除非表字段,SQL关键字尽量用大写代替小写。

39、循环内部不要声明变量,尤其是大变量:对象,解决办法是循环之前预定义需要声明的变量。

40、多维数组尽量不要循环嵌套赋值。

41、在可以用PHP内部字符串操作函数的情况下,不要用正则表达式。

42、foreach效率比while和for更高。

43、用i+=1代替i=i+1。符合c/c++的习惯,效率还高;

44、对global变量,应该用完就unset()掉;

45、有意忽略php关闭标签(即? >)。

46、写入或保存文件前,请先确保目录是可写的,假如不可写,输出错误信息。这会节约你很多调试时间。特别是 linux 系统中,需要处理权限,目录权限不当会导致很多很多的问题,文件也有可能无法读取等等。比如下面的例子:

<?php
  $contents = "All the content";
  $file_path = "/var/www/project/content.txt";
  file_put_contents($file_path ,$contents); 
?>


这大体上正确,但有些间接的问题,file_put_contents 可能会由于几个原因失败:

(1)父目录不存在

(2)目录存在,但不可写

(3)文件被写锁住?


  所以写文件前做明确的检查更好,正确写法如下:

<?php
$contents='测试内容';
$dir='/var/www/project';
$file_path=$dir."/content.txt";
  if(is_writable($dir)){
    file_put_contents($file_path,$contents);
  }else{
    die('目录不存在或者目录不可写!' );
  }
?>


47、不要依赖submit按钮值来检查表单提交行为,比如下面的情况:

<?php
  if($_POST['submit' ] == 'Save' ){ 
    //Save the things 
  }
?>


上面大多数情况正确,除了应用是多语言的。 'Save' 可能代表其它含义,你怎么区分它们呢,因此,不要依赖于submit按钮的值。正确写法如下:

<?php
  if($_SERVER['REQUEST_METHOD'] == 'POST' && isset($_POST['submit'])){
    //Save the things 
  }
?>



48、不要直接使用 $_SESSION 变量,举个简单的例子:

$_SESSION['username'] = $username; 或者 $username = $_SESSION['username'];



这会导致某些问题,如果在同个域名中运行了多个应用,session 变量可能会冲突,两个不同的应用可能使用同一个session key,例如,一个前端门户,和一个后台管理系统使用同一域名。对于这种情况,解决方案如下,使用应用相关的key和一个包装函数:

<?php
define('APP_ID','abc_corp_ecommerce');
function session_get($key){
  $k=APP_ID.'.'.$key;
 if(isset($_SESSION[$k])){
   return $_SESSION[$k];
 }
 return false;
}
function session_set($key,$value){
 $k=APP_ID.'.'.$key;
 $_SESSION[$k]=$value;
 return true;
}?>



49、将你的工具函数封装到类中,假如你在某文件中定义了很多工具函数,如下:

<?php
function utility_a(){
    //This function does a utility thing like string processing
}
function utility_b(){
    //This function does nother utility thing like database processing
}
function utility_c(){
    //This function is ..
}?>


但这些函数的使用分散到应用各处,那么你可以将他们封装到某个类中:

<?php
class Utility {
    public static function utility_a(){}
    public static function utility_b(){}
    public static function utility_c(){}
}?>


调用方法如:$a=Utility::utility_a(); 或者 $b=Utility::utility_b();

这样做的好处是,如果php内建有同名的函数,这样就可以避免冲突,维护起来也相当容易。

50、使用array_map快速处理数组,比如说你想 trim 数组中的所有元素,新手可能会:

<?php
  foreach($arr as $c => $v) { 
    $arr[$c] = trim($v); 
  }
?>


但和上面的比起来使用 array_map 更简单,比如:

<?php
  $arr = array_map('trim',$arr); 
?>


这会为$arr数组的每个元素都申请调用trim函数,另一个类似的函数是 array_walk,具体用法请查阅文档学习更多技巧.

51、使用 php filter 验证数据,你肯定曾使用过正则表达式验证 email ,ip地址等,可以尝试使用 php内置的 filter 扩展来完成相关验证和检查输入。

52、确保你的脚本由始至终都使用单一的数据库连接,在开始处正确的打开连接,使用它直到结束,最后关闭它,像下面这种在函数中打开连接是非常糟糕的:

<?php
function add_to_cart() {
    $db = new Database ();
    $db->query("INSERT INTO cart .....");
}
function empty_cart() {
    $db = new Database ();
    $db->query("DELETE FROM cart .....");
}?>


以上事例因为创建连接需要时间和占用内存,所以会拖慢应用的速度。数据库的链接最好使用单例模式。


----------------------------------------------------------------

【PHP代码优化技巧】

PHP代码优化

1.如果一个方法能被静态,那就声明他为静态的,速度可提高 1/4;

2.echo的效率高于print,因为 echo没有返回值,print返回一个整型 ;

3.在循环之前设置循环的最大次数,而非在在循环中 ;

4.销毁变量去释放内存,特别是大的数组 ;

5.避免使用像__get, __set, __autoload等魔术方法 ;//程序设计

6.requiere_once() 比较耗资源 ;

7.在includes 和requires中使用绝对路径,这样在分析路径花的时间更少 ;

8.如果你需要得sexinsex到脚本执行时的时间, $_SERVER['REQUSET_TIME']优于time();

9.能使用字符处理函数的,尽量用他们,因为效率高于正则 ;//php100.com

10.str_replace 字符替换比正则替换 preg_replace快,但strtr 比str_replace又快 1/4;

11.如果一个函数既能接受数组又能接受简单字符做为参数,例如字符替换,并且参数列表不是太长,可以考虑多用一些简洁的替换语句,一次只替换一个字符,而不是接受数组做为查找和替换参数。大事化小, 1+1>2;

12.用@ 掩盖错误会降低脚本运行速度 ;

13.$row['id'] 比$row[id]速度快 7倍,建议养成数组键加引号的习惯 ;

14.错误信息很有用;

15.在循环里别用函数,例如 For($x=0; $x < count($array); $x), count() 函数在外面先计算 ;

16.在方法里建立局部变量速度最快, 97xxoo几乎和在方法里调用局部变量一样快 ;

17.建立一个全局变量要比局部变量要慢 2倍;

18.建立一个对象属性(类里面的变量 )例如($this->prop++) 比局部变量要慢 3倍;

19.建立一个未声明的局部变量要比一个初始化的局部变量慢 9-10倍;

20.声明一个未被任何一个函数使用过的全局变量也会使性能降低 (和声明相同数量的局部变量一样 ),PHP 可能去检查这个全局变量是否存在 ;

21.方法的性能和在一个类里面定义的方法的数目没有关系,因为我添加 10个或多个方法到测试的类里面 (这些方法在测试方法的前后 )后性能没什么差异;

22.在子类里方法的性能优于在基类中 ;//PHP100中文网

23.只调用一个参数并且函数体为空的函数运行花费的时间等于 7-8次$localvar++ 运算,而一个类似的方法 (类里的函数)运行等于大约 15次$localvar++ 运算;

24.Surrounding your string by ‘ instead of ” will make things interpret a little faster since php looks for variables inside “…” but not inside ‘…’. Of course you can only do this when you don’t need to have variables in the string.

25.当输出字符串时用逗号代替点分割更快些。注意:这只对 echo起作用,这个函数能接受一些字符串作为参数 ;

26.在apache 服务器里一个 php脚本页面比相应的HTML静态页面生成至少要多花 2-10倍的时间,建议多用些静态 HTML页面和少量的脚步;

27.除非你的安装了缓存,不然你的 php脚本每次被访问都需要被重编译。建议安装个 php缓存程序,这样通过去除一些重复的编译来很明显的提高你 20-100%的性能;

28.建议用memcached ,高性能的分布式内存对象缓存系统,提高动态网络应用程序性能,减轻数据库的负担 ;

29.使用ip2long() 和long2ip()函数把 IP地址转成整型存放进数据库而非字符型。这几乎能降低 1/4的存储空间。同时可以很容易对地址进行排序和快速查找 ;

30.使用checkdnsrr() 通过域名存在性来确认部分 email地址的有效性,这个内置函数能保证每一个的域名对应一个 IP地址;// 程序设计

31.如果你在使用php5和 mysql4.1以上的版本,考虑使用mysql_*的改良函数 mysqli_*;

32.试着喜欢使用三元运算符 (?:);

33.在你想在彻底重做你的项目前,看看 PEAR有没有你需要的。PEAR是个巨大的资源库,很多 php开发者都知道;

34.使用highlight_file() 能自动打印一份很好格式化的页面源代码的副本 ; //程序设计

35. 使用error_reporting(0) 函数来预防潜在的敏感信息显示给用户。理想的错误报告应该被完全禁用在 php.ini文件里。可是如果你在用一 个共享的虚拟主机,php.ini你不能修改,那么你最好添加 error_reporting(0)函数,放在每个脚本文件的第一行 (或用 require_once() 来加载)这能有效的保护敏感的 SQL查询和路径在出错时不被显示 ;

36.使用 gzcompress() 和gzuncompress()对容量大的字符串进行压缩 (解压) 在存进(取出 )数据库时。这种内置的函数使用 gzip算法能压缩到90%;

37.通过参数变量地址得引用来使一个函数有多个返回值。你可以在变量前加个“ &”来表示按地址传递而非按值传递 ;

用户评论:

我的名片

姓名:李鹏

职业:PHP开发工程师

现居:广东省-深圳市

Email:lipeng4203@163.com