使用PHP编程的最大好处是学习这种编程语言非常容易以及其丰富的库即使对需要使用的函数不是十分了解我们也能够猜测出如何完成一个特定的任务
尽管PHP非常简单易学但我们仍然需要花费一点时间来学习PHP的一些编程技巧尤其是与性能和内存占用相关的技巧在PHP中有许多小技巧能够使我们减少内存的占用并提高应用程序的性能在本篇文章中我们将对PHP应用程序的分析如何改变脚本代码以及比较优化前后的各种参数值进行简要的介绍
通过在程序中设置计时的程序并反复执行这些代码我们可以获得有关程序执行速度的一组数据这些数据可以可以用来发现程序中的瓶颈以及如何进行优化提高应用程序的性能
也许读者曾经听说过PEAR库吧我们将使用PEAR库创建在分析时需要使用的例子这也是对现有的代码进行分析的最简单的方法它使我们无需使用商用产品就能对代码进行分析
我们要使用的库的名字是PEAR::Benchmark它对于对代码进行分析和性能测试非常有用这个库提供一个名字为Benchmark_Timer()的类能够记录一个函数调用和下一个函数调用之间的时间在对代码的性能进行测试时我们可以得到一个详细的脚本执行结果它非常简单如下所示
include_once(Benchmark/Timerphp);
$bench = new Benchmark_Timer;
$bench>start();
$bench>setMarker(Start of the script);
// 现在处于睡眠状态几分钟
sleep();
$bench>stop();
// 从计时器中获得分析信息
print_r($bench>getProfiling());
?>
上面代码执行后的输出如下所示
Array
(
[] => Array
(
[name] => Start
[time] =>
[diff] =>
[total] =>
)
[] => Array
(
[name] => Start of the script
[time] =>
[diff] => E
[total] => E
)
[] => Array
(
[name] => Stop
[time] =>
[diff] =>
[total] =>
)
)
上面的数字似乎是一组杂乱无章的数字但如果程序的规模更大这些数字就十分地有用了
也许广大读者也能猜测到数组的第一个表目是实际调用Benchmark_Timer()类的方法例如
$bench>start()$bench>setMarker()和$bench>stop()与这些表目有关的数字是相当简单的
现在我们来仔细地研究这些数字
[] => Array
(
[name] => Start
[time] =>
[diff] =>
[total] =>
)
time表目指的是何时对Benchmark_Timer()的start()方法调用的UNIX的timestampdiff表目表示这次调用和上次调用之间的时间间隔由于这里没有上一次因此显示出了一个破折号total表目指的是自测试开始到这一特定的调用之前代码运行的总的时间下面我们来看看下一个数组的输出
[] => Array
(
[name] => Start of the script
[time] =>
[diff] => E
[total] => E
)
从上面的数字我们可以看出在调用$bench>start()之后程序运行了E秒(也就是
秒)后开始调用$bench>setMarker()
一次真实的性能测试经历
尽管上面的例子不错但在对于决定如何优化你的站点代码设计方面它真的不能算是一个好例子下面我将用我自己作为网站技术人员的一段亲身经历来说明如何解决性能方面存在的问题
我并不大理解网站使用的代码因为它是根据特殊的需求历经多年开发而成的━━其中的一个模块包括网站转换代码另一个模块记录网站的使用情况其他的模块也各有各的作用我和网站的主要开发者都意识到网站的代码需要优化但又不清楚问题出在哪儿
为了尽快地完成任务我开始研究网站的主要脚本代码并在全部脚本代码以及其包含文件中添加了一些
$bench>setMarker()命令然后分析$bench>getProfiling()的输出并对得到的结果大吃一惊原来问题出在一个与获得特定语言名字(例如en代表english)的转换代码的函数调用中该函数在每个页面上都会被使用数百次每次调用该函数时脚本代码都会对一个MySQL数据库进行查询从一个数据库表中获得真正的语言名字
于是我们这一类的信息创建了一个缓沖系统经过短短天时间的工作我们使系统的性能得到了很大的提高第一周内页面的浏览量也因此而增加了%当然了这只是一个有关分析代码能够提高互联网应用或互联网网站性能的例子
性能测试函数调用
在分析一个脚本或网页(以及其包含文件)时尽管Benchmark_Timer()特别有用但它并不科学因为要获得分析的数据我们必须多次加载脚本而且它也不是针对某个类或函数调用的
PEAR::Benchmark库中的另一个被称作Benchmark_Iterator的类能够很好地解决这一个问题它能够针对特定的函数或类的方法显示其分析信息它的用途是能够能够从测试中获得一致的结果因为我们知道如果运行一段脚本一次其运行时间为秒并不意味着它每次的运行时间总是秒
In any case lets see some examples:
// 连接数据库的代码
include_once(DBphp);
$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/Iteratephp;
$bench = new Benchmark_Iterate;
// 运行getDate函数次
$bench>run( getCreatedDate );
// 打印分析信息
print_r($bench>get());
?>
运行上面的代码能够产生与下面相似的结果
Array
(
[] =>
[] =>
[] =>
[] =>
[] =>
[] =>
[] =>
[] =>
[] =>
[] =>
[mean] =>
[iterations] =>
)
上面的这些数字很好理解mean条目表示getCreatedDate()函数次运行的平均时间在进行实际测试时应该至少运行次但这个例子得出的结果已经足够说明问题了
结束语
希望广大读者能够通过本篇文章掌握如何迅速地对PHP代码进行分析的基本方法在这里我还还要提醒广大读者的是对代码进行分析不是一件简单的事儿因为我们必须掌握大量的有关该种语言的特性在代码中添加计时用的代码有助于找出运行速度缓慢的函数利用多次重复的方法使我们能够发现对代码进行正确优化的方法