PHP应用程序的性能优化 /@3+zpaw X
"3\RJ?eW:S
使用PHP编程的最大好处是学习这种编程语言非常容易以及其丰富的库。即使对需要使用的函数不是十分了解,我们也能够猜测出如何完成一个特定的任务。 30DpIkf
P?9CBhN
尽管PHP非常简单易学,但我们仍然需要花费一点时间来学习PHP的一些编程技巧,尤其是与性能和内存占用相关的技巧。在PHP中,有许多小技巧能够使我们减少内存的占用,并提高应用程序的性能。在本篇文章中,我们将对PHP应用程序的分析、如何改变脚本代码以及比较优化前后的各种参数值进行简要的介绍。 EHzZ9zH\
'/sc `(`:0
通过在程序中设置计时的程序,并反复执行这些代码,我们可以获得有关程序执行速度的一组数据,这些数据可以可以用来发现程序中的瓶颈,以及如何进行优化,提高应用程序的性能。 m9L+|r
H~ks"D1
也许读者曾经听说过PEAR库吧。我们将使用PEAR库创建在分析时需要使用的例子,这也是对现有的代码进行分析的最简单的方法,它使我们无需使用商用产品就能对代码进行分析。 M<ad>M
T^sxR4F
我们要使用的库的名字是PEAR::Benchmark,它对于对代码进行分析和性能测试非常有用。这个库提供一个名字为Benchmark_Timer()的类,能够记录一个函数调用和下一个函数调用之间的时间。在对代码的性能进行测试时,我们可以得到一个详细的脚本执行结果,它非常简单,如下所示: YvYav d
>F+:ej
include_once("Benchmark/Timer.php"); o8s&n3mY}y
$bench = new Benchmark_Timer; uFok'3!g7%
@J r
$bench-> start(); DVB:8"Bu
$bench-> setMarker('Start of the script'); (S2<6Nm8
$hKgTf?
// 现在处于睡眠状态几分钟 \&TTe8
sleep(5); Lvp/} /H/
PU'v o4
$bench-> stop(); OW-+23)sj
F)gL=6h
// 从计时器中获得分析信息 dt5gQ9(B
print_r($bench-> getProfiling()); ED?s[K
?> sm_:M| [D
U!e4_JBR'
W2<X 5'
上面代码执行后的输出如下所示: I?fE=2}9
:lE7v~!Z
3zl!x
Array _p_F v>>:
( 3/ [=
[0] => Array #e|eWi>
( iEU(1?m2-
[name] => Start Etl7V
[time] => 1013214253.05751200 ?BLOc;I&a
[diff] => - 26Yg?:kP
[total] => 0 #t/Q4X
+
) bTiw?i+6Dv
TM"-X\e~{
[1] => Array <=zGaU,
( #zy%B
[name] => Start of the script 0)P18n"$
[time] => 1013214253.05761100 Fx@
{]
[diff] => 9.8943710327148E-05 :EO}uP2
[total] => 9.8943710327148E-05 /C(L(X
) xJ"KR:CD>
{[s<\<~B*
[2] => Array cYp}$
( N!m%~},s//
[name] => Stop V`H#|8\i
[time] => 1013214258.04920700 r[,KE.^6~#
[diff] => 4.9915959835052 @"~\[z5
[total] => 4.9916949272156 G`
8j ^H,
) lyi}q"Kn*;
!e7vc[N
) )a}5\V
JJ+<?CeHD
[-CG&l2?L
-0]aOT--
g@U#Y#b@"
o}%fs
*
上面的数字似乎是一组杂乱无章的数字,但如果程序的规模更大,这些数字就十分地有用了。 `j(+Y
>u4e:/5]
也许广大读者也能猜测到,数组的第一个表目是实际调用Benchmark_Timer()类的方法,例如 l~=iUZW<
:rj78_e9
$bench-> start()、$bench-> setMarker()和$bench-> stop(),与这些表目有关的数字是相当简单的,现在我们来仔细地研究这些数字: 7'8O*EoB'
bo*q{@Ue
[0] => Array m!2Dk#t
C{ti>'"V
( yp4G"\hN9
0GR9opZtA
[name] => Start +/X'QB$R
Wp]EaYt2D
[time] => 1013214253.05751200 g|zK%tR_P
]S:@=9JB'
[diff] => - H|!s.
F9<OKcXH
[total] => 0 o2|(0uN'
VsmL#@E
) Ax%BnkU
L,ra=SV F
time表目指的是何时对Benchmark_Timer()的start()方法调用的UNIX的timestamp,diff表目表示这次调用和上次调用之间的时间间隔,由于这里没有上一次,因此显示出了一个破折号,total表目指的是自测试开始到这一特定的调用之前代码运行的总的时间。下面我们来看看下一个数组的输出: =I5XG"",
g\l;>
[1] => Array R#`itIYh
"a
g_
( '
EDi6
Jt)~h,68
[name] => Start of the script 9?:S:Sq
Ocb2XEF
[time] => 1013214253.05761100 "h2Ny#
|]q=D1/A
[diff] => 9.8943710327148E-05 saT9%?4-
%C)JmaQ{9
[total] => 9.8943710327148E-05 yRznP)
[s/@z*,M1
) cDx^}N!
Wk|z\OR(
从上面的数字我们可以看出,在调用$bench-> start()之后,程序运行了9.8943710327148E-05秒(也就是 w=`z!x![/
l+6\U6_)B
0.0000989秒)后开始调用$bench-> setMarker(....)。 l#"alU!<^
Dr1F|[
一次真实的性能测试经历 yRYWx` G
s]N-n?'G"
尽管上面的例子不错,但在对于决定如何优化你的站点代码设计方面,它真的不能算是一个好例子。下面我将用我自己作为网站技术人员的一段亲身经历来说明如何解决性能方面存在的问题。 j[fQs,efK
LnDj
我并不大理解网站使用的代码,因为它是根据特殊的需求,历经多年开发而成的━━其中的一个模块包括网站转换代码,另一个模块记录网站的使用情况,其他的模块也各有各的作用。我和网站的主要开发者都意识到网站的代码需要优化,但又不清楚问题出在哪儿。 QdTe!f|
AH`15k_i
为了尽快地完成任务,我开始研究网站的主要脚本代码,并在全部脚本代码以及其包含文件中添加了一些 </X"*G't
j+9
S
$bench-> setMarker()命令,然后分析$bench-> getProfiling()的输出,并对得到的结果大吃一惊,原来问题出在一个与获得特定语言名字(例如en代表english)的转换代码的函数调用中,该函数在每个页面上都会被使用数百次。每次调用该函数时,脚本代码都会对一个MySQL数据库进行查询,从一个数据库表中获得真正的语言名字。 64lEB>VNm
eTc`FXw`
于是我们这一类的信息创建了一个缓冲系统。经过短短2天时间的工作,我们使系统的性能得到了很大的提高,第一周内页面的浏览量也因此而增加了40%。当然了,这只是一个有关分析代码能够提高互联网应用或互联网网站性能的例子。 v2{O67j}
o
k~R[5W|'
性能测试函数调用 [FL I+;gY
,
.I^ekF
在分析一个脚本或网页(以及其包含文件)时,尽管Benchmark_Timer()特别有用,但它并不科学,因为要获得分析的数据我们必须多次加载脚本,而且它也不是针对某个类或函数调用的。 Q7r,5w&cm
(HI%C@e9
PEAR::Benchmark库中的另一个被称作Benchmark_Iterator的类能够很好地解决这一个问题,它能够针对特定的函数或类的方法,显示其分析信息。它的用途是能够能够从测试中获得一致的结果,因为我们知道,如果运行一段脚本一次,其运行时间为10秒,并不意味着它每次的运行时间总是10秒。 _Pkh`}W:
p5l$On
In any case, let's see some examples: ?a%i|Z7!
bw\=F_>L
// 连接数据库的代码 (Pd>*G\
include_once("DB.php"); zl\#n:|
$dsn = array( d]3sC
'phptype' => 'mysql', sJoi fl
7
'hostspec' => 'localhost', !d\GD8|4
'database' => 'database_name', #+
'@/5{ n
'username' => 'user_name', m3!M L>nLt
'password' => 'password' GU3/s&9
); 0f^.zt{T
$dbh = DB::connect($dsn); }L!`K"^O&
^rwSbM$
function getCreatedDate($id) lc-|Q#$3$
{ Bs?F*,zDJ
global $dbh; |esjhf}H>v
fO^6q1a
> $stmt = "SELECT created_date FROM users WHERE id=$id"; Bx-,"Z \
// 在这里使用PEAR::DB zfb _ )
$created_date = $dbh-> getOne($stmt); c0&'rxi(B
if ((PEAR::isError($created_date)) || 6t:c]G'J
(empty($created_date))) { 'I]"=O,
return false; ^ kvH/ Y&
} else { MjB[5:s
return $created_date; >e;STU
} Jt6J'MOq
} bFezTl{M
Q~JKKq
include_once 'Benchmark/Iterate.php'; <y!r~?
$bench = new Benchmark_Iterate; UwkX[u
^4pKsO3ul
// 运行getDate函数10次 &|}IBu :T
$bench-> run(10, 'getCreatedDate', 1); L_"(A
#H:
yrAzD=
// 打印分析信息 Uob |Q=MQ
print_r($bench-> get()); ATM:As:<@
?> HY;?z`=
%uVJLz
Lc<xgN+cJ
g@i>R>
4D$sFR|?t
运行上面的代码能够产生与下面相似的结果: 2?iOB6
6;frIl;
zL'IN)7MU
Array %D(prA_w
( -!,]Y10
[1] => 0.055413007736206 jHlOP,kc
[2] => 0.0012860298156738 7/_ VE
[3] => 0.0010279417037964 /}`/i(k
[4] => 0.00093603134155273 Gr$*t,ZW
[5] => 0.00094103813171387 Pmlgh&Z
[6] => 0.00092899799346924 QX.6~*m1
[7] => 0.0010659694671631 %K'*P56
[8] => 0.00096404552459717 C'/M/|=Q#
[9] => 0.0010690689086914 _SC
[10] => 0.00093603134155273 ?vn 0%e868
[mean] => 0.0064568161964417 1 {x~iZa
[iterations] => 10 ZT"|o\G^Q
) 7.
9s.*
6'Yn|A
b+].Uc
eH%L?"J~:
上面的这些数字很好理解,mean条目表示getCreatedDate()函数10次运行的平均时间。在进行实际测试时,应该至少运行1000次,但这个例子得出的结果已经足够说明问题了。 H!r
Kz
}<ONx g6Kb
结束语 l$VxE'&LQ
I.+)sB?5
希望广大读者能够通过本篇文章掌握如何迅速地对PHP代码进行分析的基本方法。在这里我还还要提醒广大读者的是,对代码进行分析不是一件简单的事儿,因为我们必须掌握大量的有关该种语言的特性。在代码中添加计时用的代码有助于找出运行速度缓慢的函数,利用多次重复的方法使我们能够发现对代码进行正确优化的方法。