Loading...

当前位置:资讯中心主页 >PHP编程 >文章内容

  • PHP应用程式的性能优化
  • 来源:作者: 发布时间:2007-12-19 18:03:55
    • 域名注册

    • 域名惊喜价格 cn域名1元注册
    • com域名39.9

      虚拟主机

    • 主机按月支付,低至19元/月
    • 超大流量,可开子站点

      VPS主机

    • 特惠VPS168元/月,4-8M独享带宽保证
    • 独立操作系统,无限开站点
    使用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代码进行分析的基本方法。在这里我还还要提醒广大读者的是,对代码进行分析不是一件简单的事儿,因为我们必须掌控大量的有关该种语言的特性。在代码中添加计时用的代码有助于找出运行速度缓慢的函数,利用多次重复的方法使我们能够发现对代码进行正确优化的方法。


  • 以上内容由 华夏名网 搜集整理,如转载请注明原文出处,并保留这一部分内容。

      “华夏名网” http://www.sudu.cn 和 http://www.bigwww.com 是成都飞数科技有限公司的网络服务品牌,专业经营虚拟主机,域名注册,VPS,服务器租用业务。公司创建于2002年,经过6年的高速发展,“华夏名网”已经成为我国一家知名的互联网服务提供商,被国外权威机构webhosting.info评价为25大IDC服务商之一。

    华夏名网网址导航: 虚拟主机 双线主机 主机 域名注册 cn域名 域名 服务器租用 酷睿服务器 vps vps主机

  • (阅读次数:12)
  • 上一篇: 用PHP将数据导入到Foxmail    下一篇: 使用PHP4中的 IntegratedTemplate类实现HTML和PHP代码分离
  • [收藏] [推荐] [评论] [打印本页] [返回上一页][关闭窗口]
  • 昵称: (为空则显示guest)
  • 评论分数: ★ ★ ★★★ ★★★★ ★★★★★
  • 评论内容:(不能超过250字,需审核后才会公布,请自觉遵守互联网相关政策法规。