采用 PEAR 来缓冲 PHP 程序 Z9$pY=8^?
"h_f-vP
PHP 世界中缓冲是一个热门的话题,因为 PHP 产生的动态页面,每次用户请求都需要重新计算,不论请求的结果是否一样,同时,PHP 每次都会编译一次脚本。这样的超负荷运转对一个流量很高的网站来说肯定难以忍受。幸运的是, Web 的结果可以缓冲,而不需要重新运行和编译脚本,商品化的产品像 ZendCache 或者开源的 Alternate PHP Cache都提供了把 PHP 脚本编译为字节代码并缓冲的办法。 ftTD-d
aOGoJCt
C
PEAR 的缓冲包提供了缓冲动态内容,数据库查询和 PHP 函数调用的框架。 p-{ 4 $W
d9:I.SA)E
就像 Perl 有 CPAN, TeX 有 CTAN,PHP 也有自己的中心资源库,存放类,库和模块。这个库称为 PEAR(PHP Extension and Add-On Repository)。 dY&v(~&;]
#~nXAs]Q
本文假设你已经安装了 PEAR 环境,如果没有的话,可以去 PHP 网站下载。 y/Y}C.IWp)
yKl^-%Uq<
H!]&"V77
PEAR 的缓冲包包含一个总体的缓冲类和几个特别的子类。缓冲类使用容器类来存贮和管理缓冲数据。 -%MXt
S8dfe~ |7:
下面是 PEAR 缓冲当前所包含的容器,以及各自的参数: /B?wn=][
aC2Vz9e
file -- file 容器在文件系统存储了缓冲的数据,是最快的容器。 01-rBto$
h<3b+*wYJC
cache_dir -- 这是容器存储文件的目录。 Nmz5:Rq
j%
7Gje[
filename_prefix -- 缓冲文件的前缀,例如:"cache_"。 lqOpADLS3
/kG?I_z
shm -- shm 容器把缓冲数据放入共享内存,基准测试显示,目前的实现下,这个容器的速度要比文件容器慢。 rtz-kQ38R
X,l7>>L{g
shm_key -- 共享内存使用的键值。 xbhHP2F|
z3i`O
La
shm_perm -- 使用共享内存数据段的权限。 Yv]vl6<
VVch%
shm_size -- 分配共享内存的大小。 i4D]>
51|s2+GG
sem_key -- 信号灯的键值。 "rLm)$I
e8 ]CB
sem_perm -- 信号灯的权限。 F]6G<6T[
'XKfKv >;
db -- PEAR 的数据库抽象层。 A"M;kzAfHM
A+M4=
dsn -- 数据库连接的 DSN 。可以参考 PEAR 的 DB 文档。 5A`>3w{3n
0Sd>*nC
cache_table -- 表的名字。 ASoBa&vX
p1niS:}j
phplib -- phplib 容器使用数据库抽象层存储缓冲。 e_ epuki
ZrEou}z(*
db_class 02;'"EmP$
YX,;z/Jw2
db_file seK;TQ3/7
33lh~+C
db_path u->[y1JY
V=+|]`
local_file D.{vuftu
==?wG!v2 h
local_path [DjlkA/Zg
\[{8E}_"^
ext/dbx -- PHP 的数据库抽象层扩展,如果像把缓冲存入数据库,可以采用这个容器。 v1yB
!%t@wQ]\hG
module `;}qjm0a
nw/g[/<;
host Zc_F"KJL
vo
}4N[]Sb
db Kn$E{ F\
<`SA>P
username 83V\O_7j
Vbp@n
password }|Q\@3&
kK}?NKqT
cache_table B^TgEr
2
oL$I(83
persistent C<a&]dN/
&?QKWxN
使用 PEAR Cache 所得到的性能提升取决于你所选择的缓冲容器,例如,把数据库的结果再次存入数据库缓冲中就显得毫无意义。 3t9+Y dNKU
*y<eK0
PEAR Cache 的函数缓冲模块能把任何函数或者方法的结果缓冲,不论是 PHP 的内置函数还是用户自定义函数,他缺省采用文件容器,把缓冲数据放入到一个叫做 'j'6x'[>]
function_cache 的目录。 THOYx :Nr;
.{t5_,P
jNX6Ct?
Cache_Function 类的构造器可以有三个可选的参数: W7|nc,i0\
WNjG/U
$container :缓冲容器的名字。 [^a7l$fmi
#B?lU"f8q^
$container_options :缓冲容器的数组参数。 sSKD"
zwQ#Yvd
$expires:缓冲对象过期的时间(秒数)。 8hww({S2
30I-E._F
普通的函数调用采用 Cache_Function 类的 call() 方法时,就能触发缓冲。调用 call() 很容易,的一个参数是函数的名字,然后是函数的参数,第二个参数是要调用函数中的第一个,依此类推,我们来看例子: qm_r~j
zp9l u B
例 1: 缓冲函数和方法的调用 :yJ#yad
jt6_1^
// 调用 PEAR Cache 的函数缓冲。 1
Lg {l
&k*oG:J3
<?php ImB5F'HI$
require_once 'Cache/Function.php'; ^"lEa-g&
^2BiMH3j
// 定义一些类和函数。 E]vox~xK>
S3HyB
b
class foo { )Dhx6xM[a
function bar($test) { ~FAk4z=Ed
echo "foo::bar($test)<br>"; =YO<.(Lu
} NoF|j57?u'
} B)DuikV.D
nvQX)Xf
class bar { R!"`Po
function foobar($object) { I+Yq",{%
echo '$'.$object.'->foobar('.$object.') c]k+ Sx&}
'; HI:1Voy
} mmjWLrhlu
} L"1AC&~u
=`(W^&|
$bar = new bar; G
9 &,`
7ieAd/:_
function foobar() { w?"M
echo 'foobar()'; 8.wtv5eZ
} 4!ZT_q
>@G"*le*)
// 取得 Cache_Function 对象 y~OP9Tg
PVxu8n
$cache = new Cache_Function(); <M5fk?n,|
6,1oLvU
// 对 foo 类的静态函数 bar() 作缓冲(foo::bar())。 pfc"^Gi8
$cache->call('foo::bar', 'test'); ?)<zzL",
\TzBu?,v8
// $bar->foobar() #:Q\
$cache->call('bar->foobar', 'bar'); QS4~":D/C
S~m8j|3K
$cache->call('foobar'); nRX'J5Q
m<
?> GHi'ek <?^
c*<BU6y
"ig)7X+Wz|
~A%+oa*2~
下面我们采用 Cache_Output 来把输出作缓冲: ?c"iV
^g2Vz4u
例子 2: 缓冲脚本的输出 M'X,7hZ
@!ja/Y^
// 加载 PEAR Cache 的输出缓冲 !YO'u'4<aK
Mg}/gO%o
<?php gE*7[*2?t
require_once 'Cache/Output.php'; zFYzus`>
'O2/PU2_
$cache = new Cache_Output('file', array('cache_dir' => '.') ); `{IL.9M!f
`25<;@
// 计算要缓冲页面的标记,我们假定页面的缓冲取决于 )3|a_
// URL, HTTP GET 和 POST 变量以及 cookies。 LtUw
u+6L>7t88I
$cache_id = $cache->generateID(array('url' => $REQUEST_URI, 'post' => $HTTP_POST_VARS, 'cookies' => $HTTP_COOKIE_VARS) ); D^s#pOZS
&>Z;>6J,
// 查询缓冲 [\fwnS_1
E}0g
if ($content = $cache->start($cache_id)) { L|b[6[XTHL
2*gB ~Jn4
// 缓冲命中 p,(W?.ZDN?
echo $content; c*R\fQd
die(); Ed-3-vJej6
} g#1Y4
]TtID4qL
// 缓冲丢失 muK.x7zyl
e6 <9`Xg
// -- 在这里插入内容产生代码 -- TZg1,Z
t1yfSStp
// 把页面存入缓冲 >@a7Zzl0H
echo $cache->end(); F_/ra?WVH
?> 9@Cu5U]
eQ[}ALIq
利用 Cache_Output 类,很容易把一个动态的数据库驱动的网站应用转化为静态,从而极大的提升站点的性能。 ;jPiD`Kyv
>lJTS t5{
越来越多的站点在采用 GZIP 压缩 HTML 内容,这样减少了服务器的带宽消耗,对于使用 Modem 上网的用户来说也能受益不少。 eqOT@~H
TB<$9FCHK
Cache_OutputCompression 扩展了 Cache_Output 类的功能,他把 GZIP 压缩的 HTML 内容进行缓冲,从而节省了 CPU 压缩的时间。