PHP代码优化

17 篇文章 0 订阅
张存超php技术博客
    优化PHP代码的40条建议
    1.如果一个方法可静态化,就对它做静态声明。速率可提升至4倍。
    2.echo 比 print 快。
    3.使用echo的多重参数(译注:指用逗号而不是句点)代替字符串连接。
    4.在执行for循环之前确定最大循环数,不要每循环一次都计算最大值。
    5.注销那些不用的变量尤其是大数组,以便释放内存。
    6.尽量避免使用__get,__set,__autoload。
    7.require_once()代价昂贵。
    8.在包含文件时使用完整路径,解析操作系统路径所需的时间会更少。
    9.如果你想知道脚本开始执行(译注:即服务器端收到客户端请求)的时刻,使用$_SERVER[‘REQUEST_TIME’]要好于time()。
    10.函数代替正则表达式完成相同功能。
    11.str_replace函数比preg_replace函数快,但strtr函数的效率是str_replace函数的四倍。
    12.如果一个字符串替换函数,可接受数组或字符作为参数,并且参数长度不太长,那么可以考虑额外写一段替换代码,使得每次传递参数是一个字符,而不是只写一行代码接受数组作为查询和替换的参数。
    13.使用选择分支语句(译注:即switch case)好于使用多个if,else if语句。
    14.用@屏蔽错误消息的做法非常低效。
    15.打开apache的mod_deflate模块。
    16.数据库连接当使用完毕时应关掉。
    17.$row[‘id’]的效率是$row[id]的7倍。
    18.错误消息代价昂贵。
    19.尽量不要在for循环中使用函数,比如for ($x=0; $x < count($array); $x)每循环一次都会调用count()函数。
    20.在方法中递增局部变量,速度是最快的。几乎与在函数中调用局部变量的速度相当。
    21.递增一个全局变量要比递增一个局部变量慢2倍。
    22.递增一个对象属性(如:$this->prop++)要比递增一个局部变量慢3倍。
    23.递增一个未预定义的局部变量要比递增一个预定义的局部变量慢9至10倍。
    24.仅定义一个局部变量而没在函数中调用它,同样会减慢速度(其程度相当于递增一个局部变量)。PHP大概会检查看是否存在全局变量。
    25.方法调用看来与类中定义的方法的数量无关,因为我(在测试方法之前和之后都)添加了10个方法,但性能上没有变化。
    26.派生类中的方法运行起来要快于在基类中定义的同样的方法。
    27.调用带有一个参数的空函数,其花费的时间相当于执行7至8次的局部变量递增操作。类似的方法调用所花费的时间接近于15次的局部变量递增操作。
    28.用单引号代替双引号来包含字符串,这样做会更快一些。因为PHP会在双引号包围的字符串中搜寻变量,单引号则不会。当然,只有当你不需要在字符串中包含变量时才可以这么做。
    29.输出多个字符串时,用逗号代替句点来分隔字符串,速度更快。注意:只有echo能这么做,它是一种可以把多个字符串当作参数的“函数”(译注:PHP手册中说echo是语言结构,不是真正的函数,故把函数加上了双引号)。
    30.Apache解析一个PHP脚本的时间要比解析一个静态HTML页面慢2至10倍。尽量多用静态HTML页面,少用脚本。
    31.除非脚本可以缓存,否则每次调用时都会重新编译一次。引入一套PHP缓存机制通常可以提升25%至100%的性能,以免除编译开销。
    32.尽量做缓存,可使用memcached。memcached是一款高性能的内存对象缓存系统,可用来加速动态Web应用程序,减轻数据库负载。对运算码 (OP code)的缓存很有用,使得脚本不必为每个请求做重新编译。
    33.当操作字符串并需要检验其长度是否满足某种要求时,你想当然地会使用strlen()函数。此函数执行起来相当快,因为它不做任何计算,只返回在zval 结构(C的内置数据结构,用于存储PHP变量)中存储的已知字符串长度。但是,由于strlen()是函数,多多少少会有些慢,因为函数调用会经过诸多步骤,如字母小写化(译注:指函数名小写化,PHP不区分函数名大小写)、哈希查找,会跟随被调用的函数一起执行。在某些情况下,你可以使用isset() 技巧加速执行你的代码。
    (举例如下)
    if (strlen($foo) < 5) { echo “Foo is too short”$$ }
    (与下面的技巧做比较)
    if (!isset($foo{5})) { echo “Foo is too short”$$ }
    调用isset()恰巧比strlen()快,因为与后者不同的是,isset()作为一种语言结构,意味着它的执行不需要函数查找和字母小写化。也就是说,实际上在检验字符串长度的顶层代码中你没有花太多开销。
    34.当执行变量$i的递增或递减时,$i++会比++$i慢一些。这种差异是PHP特有的,并不适用于其他语言,所以请不要修改你的C或Java代码并指望它们能立即变快,没用的。++$i更快是因为它只需要3条指令(opcodes),$i++则需要4条指令。后置递增实际上会产生一个临时变量,这个临时变量随后被递增。而前置递增直接在原值上递增。这是最优化处理的一种,正如Zend的PHP优化器所作的那样。牢记这个优化处理不失为一个好主意,因为并不是所有的指令优化器都会做同样的优化处理,并且存在大量没有装配指令优化器的互联网服务提供商(ISPs)和服务器。
    35.并不是事必面向对象(OOP),面向对象往往开销很大,每个方法和对象调用都会消耗很多内存。
    36.并非要用类实现所有的数据结构,数组也很有用。
    37.不要把方法细分得过多,仔细想想你真正打算重用的是哪些代码?
    38.当你需要时,你总能把代码分解成方法。
    39.尽量采用大量的PHP内置函数。
    40.如果在代码中存在大量耗时的函数,你可以考虑用C扩展的方式实现它们。
    41.评估检验(profile)你的代码。检验器会告诉你,代码的哪些部分消耗了多少时间。Xdebug调试器包含了检验程序,评估检验总体上可以显示出代码的瓶颈。
    42.mod_zip可作为Apache模块,用来即时压缩你的数据,并可让数据传输量降低
    1,在函数中,传递数组时
    使用 return 比使用 global 要高效
    比如
    function userloginfo($usertemp){
    $detail=explode("|",$usertemp);
    return $detail;
    }
    $login=userloginfo($userdb);
    比
    function userloginfo($usertemp){
    global $detail;
    $detail=explode("|",$usertemp);
    }
    userloginfo($userdb);
    要高效
    2,(这个代码用于得到程序目录对应的网址,推荐使用)
    $urlarray=explode('/',$HTTP_SERVER_VARS['REQUEST_URI']);
    $urlcount=count($urlarray);unset($urlarray[$urlcount-1]);
    $ofstarurl='http://'.$HTTP_SERVER_VARS['HTTP_HOST'].implode('/',$urlarray);
    这段代码比
    $pre_urlarray=explode('/',$HTTP_SERVER_VARS['HTTP_REFERER']);
    $pre_url=array_pop($pre_urlarray);
    要高效
    3,在循环中判断时,数值判断使用恒等要比等于高效
    $a=2;$b=2;
    比如
    if($a==$b)$c=$a;
    比
    if($a===$b)$c=$a;
    高效
    4,mysql 查询时尽量使用where in 少用 limit
    limit查多记录的前几条, 速度很快, 但是查询最面几条就会慢
    使用in .在查询连续性记录,非常快, 非连续性记录第一次运行会稍微慢一点,但是之后将比较快!
    5,NT服务器数据操作稳定性不及unix/linux
    6,输出前使用尽量使用 ob_start(); 可以加快输出速度,适用NT或nuli/linux,对unlix类服务器 如果使用 ob_start('ob_gzhandler');输出效率将更高
    7,判断的时候尽量使用if($a==他的值) 否定的时候尽量使用if(empty($a)),因为这样程序运行更快速
    8,使用不等时 != 与 <> 效率相当
    9,个人经验得 使用 $a="11111111111111"; 的效率和 $a='11111111111111'; 相当.并不象书本说的相差很大
    10,使用规范的SQL语句, 会有利于MySQL的解析
    11,使用
    if($online){
    $online1=$online;
    setcookie('online1',$online,$cookietime,$ckpath,$ckdomain,$secure);
    }
    COOKIE将马上生效
    使用
    if($online)
    setcookie('online1',$online,$cookietime,$ckpath,$ckdomain,$secure);
    COOKIE需要再刷新一次才能生效
    12,使用
    $handle=fopen($filename,wb);
    flock($handle,LOCK_SH);
    $filedata=fread($handle,filesize($filename));
    fclose($handle);
    比
    file($filename);
    无论在速度还是稳定上都要优秀
    13,截断字符串优化函数(可避免?字符出现)
    function substrs($content,$length) {
    if(strlen($content)>$length){
    $num=0;
    for($i=0;$i<$length-3;$i++) {
    if(ord($content[$i])>127)$num++;
    }
    $num%2==1 ? $content=substr($content,0,$length-4):$content=substr($content,0,$length-3);
    $content.=' ……';
    }
    return $content;
    }
    比如$newarray[1]=substrs($newarray[1],25);
    14,程序中屏蔽大小写
    for ($asc=65;$asc<=90;$asc++)
    { //strtolower() 此函数在一些服务器会产生乱码!
    if (strrpos($regname,chr($asc))!==false)
    {
    $error="为了避免用户名混乱,用户名中禁止使用大写字母,请使用小写字母";
    $reg_check=0;
    }
    }
    15,不使用 file();和不使用 fget();(不稳定或速度慢) 取一数组函数
    function openfile($filename,$method="rb")
    {
    $handle=@fopen($filename,$method);
    @flock($handle,LOCK_SH);
    @$filedata=fread($handle,filesize($filename));
    @fclose($handle);
    $filedata=str_replace("\n","\n<ofstar:>",$filedata);
    $filedb=explode("<ofstar:>",$filedata);
    //array_pop($filedb);
    $count=count($filedb);
    if($filedb[$count-1]==''){unset($filedb[$count-1]);}
    return $filedb;
    }
    //这个函数虽然代码比较多,不过在速度和稳定性上优势很大!
    PHP应用程序的性能优化
    放出来,不知有多少人会去看完,但好东西一定要分享出来自己才觉得有价值.
    by soola
    使用PHP编程的最大好处是学习这种编程语言非常容易以及其丰富的库。即使对需要使用的函数不是十分了解,我们也能够猜测出如何完成一个特定的任务。
    尽管PHP非常简单易学,但我们仍然需要花费一点时间来学习PHP的一些编程技巧,尤其是与性能和内存占用相关的技巧。在PHP中,有许多小技巧能够使我们减少内存的占用,并提高应用程序的性能。在本篇文章中,我们将对PHP应用程序的分析、如何改变脚本代码以及比较优化前后的各种参数值进行简要的介绍。
    通过在程序中设置计时的程序,并反复执行这些代码,我们可以获得有关程序执行速度的一组数据,这些数据可以可以用来发现程序中的瓶颈,以及如何进行优化,提高应用程序的性能。
    也许读者曾经听说过PEAR库吧。我们将使用PEAR库创建在分析时需要使用的例子,这也是对现有的代码进行分析的最简单的方法,它使我们无需使用商用产品就能对代码进行分析。
    我们要使用的库的名字是PEAR::Benchmark,它对于对代码进行分析和性能测试非常有用。这个库提供一个名字为Benchmark_Timer()的类,能够记录一个函数调用和下一个函数调用之间的时间。在对代码的性能进行测试时,我们可以得到一个详细的脚本执行结果,它非常简单,如下所示:
    <?php
    include_once("Benchmark/Timer.php");
    $bench = new Benchmark_Timer; $bench->start();
    $bench->setMarker('Start of the script'); 现在处于睡眠状态几分钟
    sleep(5); $bench->stop(); // 从计时器中获得分析信息
    print_r($bench->getProfiling());
    >
    上面代码执行后的输出如下所示:
    Array
    (
    [0] => Array
    (
    [name] => Start
    [time] => 1013214253.05751200
    [diff] => -
    [total] => 0
    )
    [1] => Array
    (
    [name] => Start of the script
    [time] => 1013214253.05761100
    [diff] => 9.8943710327148E-05
    [total] => 9.8943710327148E-05
    )
    [2] => Array
    (
    [name] => Stop
    [time] => 1013214258.04920700
    [diff] => 4.9915959835052
    [total] => 4.9916949272156
    )
    )
    上面的数字似乎是一组杂乱无章的数字,但如果程序的规模更大,这些数字就十分地有用了。
    也许广大读者也能猜测到,数组的第一个表目是实际调用Benchmark_Timer()类的方法,例如
    $bench->start()、$bench->setMarker()和$bench->stop(),与这些表目有关的数字是相当简单的,现在我们来仔细地研究这些数字:
    [0] => Array
    (
    [name] => Start
    [time] => 1013214253.05751200
    [diff] => -
    [total] => 0
    )
    time表目指的是何时对Benchmark_Timer()的start()方法调用的UNIX的timestamp,diff表目表示这次调用和上次调用之间的时间间隔,由于这里没有上一次,因此显示出了一个破折号,total表目指的是自测试开始到这一特定的调用之前代码运行的总的时间。下面我们来看看下一个数组的输出:
    [1] => Array
    (
    [name] => Start of the script
    [time] => 1013214253.05761100
    [diff] => 9.8943710327148E-05
    [total] => 9.8943710327148E-05
    )
    从上面的数字我们可以看出,在调用$bench->start()之后,程序运行了9.8943710327148E-05秒(也就是
    0.0000989秒)后开始调用$bench->setMarker(……)。
    一次真实的性能测试经历
    尽管上面的例子不错,但在对于决定如何优化你的站点代码设计方面,它真的不能算是一个好例子。下面我将用我自己作为网站技术人员的一段亲身经历来说明如何解决性能方面存在的问题。
    我并不大理解网站使用的代码,因为它是根据特殊的需求,历经多年开发而成的━━其中的一个模块包括网站转换代码,另一个模块记录网站的使用情况,其他的模块也各有各的作用。我和网站的主要开发者都意识到网站的代码需要优化,但又不清楚问题出在哪儿。
    为了尽快地完成任务,我开始研究网站的主要脚本代码,并在全部脚本代码以及其包含文件中添加了一些$bench->setMarker()命令,然后分析$bench->getProfiling()的输出,并对得到的结果大吃一惊,原来问题出在一个与获得特定语言名字(例如en代表english)的转换代码的函数调用中,该函数在每个页面上都会被使用数百次。每次调用该函数时,脚本代码都会对一个MySQL数据库进行查询,从一个数据库表中获得真正的语言名字。
    于是我们这一类的信息创建了一个缓冲系统。经过短短2天时间的工作,我们使系统的性能得到了很大的提高,第一周内页面的浏览量也因此而增加了40%。当然了,这只是一个有关分析代码能够提高互联网应用或互联网网站性能的例子。
    性能测试函数调用
    在分析一个脚本或网页(以及其包含文件)时,尽管Benchmark_Timer()特别有用,但它并不科学,因为要获得分析的数据我们必须多次加载脚本,而且它也不是针对某个类或函数调用的。
    PEAR::Benchmark库中的另一个被称作Benchmark_Iterator的类能够很好地解决这一个问题,它能够针对特定的函数或类的方法,显示其分析信息。它的用途是能够能够从测试中获得一致的结果,因为我们知道,如果运行一段脚本一次,其运行时间为10秒,并不意味着它每次的运行时间总是10秒。
    In any case, let's see some examples:
    // 连接数据库的代码
    include_once("DB.php");
    $dsn = array(
    'phptype' => 'mysql',
    'hostspec' => 'localhost',
    'database' => 'database_name',
    'username' => 'user_name',
    'password' => 'password'
    );
    $dbh = DB::connect($dsn); function getCreatedDate($id)
    {
    global $dbh; >$stmt = "SELECT created_date FROM users WHERE id=$id";
    // 在这里使用PEAR::DB
    $created_date = $dbh->getOne($stmt);
    if ((PEAR::isError($created_date)) ||
    (empty($created_date))) {
    return false;
    } else {
    return $created_date;
    }
    }include_once 'Benchmark/Iterate.php';
    $bench = new Benchmark_Iterate; // 运行getDate函数10次
    $bench->run(10, 'getCreatedDate', 1);// 打印分析信息
    print_r($bench->get());
    ?>
    运行上面的代码能够产生与下面相似的结果:
    Array
    (
    [1] => 0.055413007736206
    [2] => 0.0012860298156738
    [3] => 0.0010279417037964
    [4] => 0.00093603134155273
    [5] => 0.00094103813171387
    [6] => 0.00092899799346924
    [7] => 0.0010659694671631
    [8] => 0.00096404552459717
    [9] => 0.0010690689086914
    [10] => 0.00093603134155273
    [mean] => 0.0064568161964417
    [iterations] => 10
    )
    上面的这些数字很好理解,mean条目表示getCreatedDate()函数10次运行的平均时间。在进行实际测试时,应该至少运行1000次,但这个例子得出的结果已经足够说明问题了。

张存超php技术博客

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值