转自:http://www.cnrui.cn/blog/article.asp?id=237
最近在深入了解PHP的一些开发特性,遇到了不少的问题。其中以Xcache的部署和使用中麻烦最大,为了不让一些朋友走弯路,所以在这里做一个小结。
描述
XCache 是一个又快又稳定的 PHP opcode 缓存器. 经过良好的测试并在大流量/高负载的生产机器上稳定运行. 经过(在 linux 上)测试并支持所有现行 PHP 分支的最新发布版本, 如 PHP_4_3 PHP_4_4 PHP_5_0 PHP_5_1 PHP_5_2 HEAD(6.x), 并支持线程安全/Windows. 与同类 opcode 缓存器相比更胜一筹, 比如能够快速跟进 PHP 版本. 详情请参考http://xcache.lighttpd.net/介绍.
直接一点形容Xcahce的2个特点:
- 将PHP代码进行编译(表现类似于.NET和JAVA的编译,内部机制不同),这样可以不用重复读取PHP文件,加速整个程序的运行效率;
- 可以缓存数据到内存中,直接可以方便调用(我一直很诟病PHP为什么连Application也没有),例如你的一些不会经常改动的数据,只用一次从数据库读取出来后保存到Xcache中,就不用再去使用数据库了(这个是理想环境下)。
在这里可以下载你所需要的相应版本:http://xcache.lighttpd.net/wiki/Release-1.2.2
这里要说明一下在windows下可能会遇到的问题:
不管你使用IIS或者Apache,如果使用FAST-CGI方式的话,下载XCache 1.2.2 for php 5.2.5-nts,特别注意的是必须设置C:\Windows\Temp的User或相关用户可读写的权限。
如果你windows上的Apache不是用的FAST-CGI方式,就下载XCache 1.2.2 for php 5.2.5
如何设置Windows上的Apache加载PHP的方式:
打开Apache目录下的conf/httpd.conf,找到 LoadModule php5_module
FAST-CGI方式设置:
LoadModule php5_module "D:/Program Files/php-apache/php5apache2_2_filter.dll"
非FAST-CGI方式设置:
LoadModule php5_module "D:/Program Files/php-apache/php5apache2_2.dll"
注意:在windows下使用该模式切记别装mysql-5.1.26-rc-win32.exe
否则会出现可以加载MYSQL,却无法使用的问题,Apache会自动挂掉!
将你的Xcache下载文件解压,然后把 php_xcache.dll 放到你的PHP的ext目录下。
比如:D:\Program Files\php-apache\ext\php_xcache.dll
接着将Xcache文件夹的xcache.ini内容拷贝到php.ini中最后面,我的配置如下:
[xcache-common]
;; 安装成 zend extension (推荐), 路径一般是 "$extension_dir/xcache.so"
;; zend_extension = /usr/local/lib/php/extensions/non-debug-non-zts-xxx/xcache.so
;; Windows 系统例子:
;;zend_extension_ts = D:\Program Files\php-apache\ext\php_xcache.dll
;; 或者您也可把 XCache 安装成 extension, 注意确保您的 extension_dir 设置正确, 并把 xcache.so 或者 php_xcache.dll 放到该目录里面
; extension = xcache.so
;; 或者 Win32 系统:
extension = php_xcache.dll
[xcache.admin]
xcache.admin.enable_auth = On
xcache.admin.user = "username"
; xcache.admin.pass = md5($您的密码)
xcache.admin.pass = "password"
[xcache]
; 这里的多数选项仅在 ini 里可以修改, 这里列出的都是默认值, 除非另外说明
; select low level shm/allocator scheme implemenation
xcache.shm_scheme = "mmap"
; 禁用: xcache.size=0
; 启用: xcache.size=64M 之类 (任意>0的值) 同时请注意您的系统 mmap 上限
xcache.size = 64M
; 建议设置为 cpu 数 (cat /proc/cpuinfo |grep -c processor)
xcache.count = 1
; 只是个参考值, 您可以放心地存储多于此数量的项目(php脚本/变量)
xcache.slots = 8K
; 缓存项目的 ttl, 0=永久
xcache.ttl = 0
; 扫描过期项目的时间间隔, 0=不扫描, 其他值以秒为单位
xcache.gc_interval = 0
; 同上, 只是针对变量缓存设置
xcache.var_size = 32M
xcache.var_count = 1
xcache.var_slots = 8K
; 默认, 允许 ini_set()
xcache.var_ttl = 0
; 最大ttl值
xcache.var_maxttl = 0
xcache.var_gc_interval = 0
; 仅测试用
xcache.test = Off
; /dev/zero 时无效
xcache.readonly_protection = Off
; 对于 *nix 系统, xcache.mmap_path 是文件路径而不是目录, (可以不存在 但是必须能创建).
; 如果您期望启用 ReadonlyProtection, 可以使用类似 "/tmp/xcache"
; 2 个 php 组不会共享同一个 /tmp/xcache
; 对于 Win32 系统, xcache.mmap_path=匿名MAP名字, 不是文件路径. 建议使用 XCache 字眼避免跟其他软件冲突
xcache.mmap_path = "zero"
; 仅用于 *nix 系统
; 设置为空(禁用) 或者类似 "/tmp/phpcore/"
; 注意该目录应该能被 php 写入文件 (跟 open_basedir 无关)
xcache.coredump_directory = ""
; 下面的项目开启后将实现PHP代码缓存
xcache.cacher = Off
xcache.stat = Off
xcache.optimizer = Off
[xcache.coverager]
; 如果 xcache.coveragedump_directory 设置为空则本设置自动为 Off
xcache.coverager = Off
; 请确保本目录能被 coverage viewer 脚本读取 (注意 open_basedir)
; 依赖于 xcache.coverager=On
xcache.coveragedump_directory = ""
Ok,重启你的web服务器(IIS 或者 Apache),查看phpinfo();,就可以看到Xcache已经被加载了。
这里要注意的一点就是Xcache只能缓存默认的一些对象,如int, string, array等,不能缓存对象,否则读取的时候就会报错。
如果你非要缓存对象的话也有办法就是将对象序列化,读取的时候再反序列化一次。
下面我写的一个Xcache的简单类:
<?php
/**
* Xcache moudle
*/
class cacheHelper{
public $prefix;
function __construct(){
if(!function_exists('xcache_get')){
exit("This application must required XCache module.");
}
}
/**
* __set
*
* @param mixed $name
* @param mixed $value
* @access public
* @return void
*/
public function __set($name, $value){
xcache_set($this->prefix.$name, $value);
}
/**
* __get
*
* @param mixed $name
* @access public
* @return mixed
*/
public function __get($name){
return xcache_get($this->prefix.$name);
}
/**
* __isset
*
* @param mixed $name
* @access public
* @return bool
*/
public function __isset($name){
return xcache_isset($this->prefix.$name);
}
/**
* __unset
*
* @param mixed $name
* @access public
* @return void
*/
public function __unset($name){
xcache_unset($this->prefix.$name);
}
}
?>
和Xcache一样,还有其他的几种缓存方案可选,如APC(Alternative PHP Cache),eAccelerator,memcahced。具体的配置和使用方法网上很多就不重复了。
下面是我的APC配置文件:
[APC]
;启用 APC 扩展
extension = php_apc.dll
;开启 APC 1 | Off
apc.enabled = 1
;可分配的共享内存个数
apc.shm_segments = 1
;每个共享内存的尺寸大小,这里是 64M
apc.shm_size = 64
;当文件的尺寸大于这个值是,将不会被缓存,默认是1M
apc.max_file_size = 10M
;当设置为1时,表示每次请求进来时,APC 都会检查文件是否被修改,如果修改了的话就会重新编译,默认是 1
apc.stat=1
apc.cache_by_default = On
;在 CLI 模式(命令行模式)下启用 APC
apc.enable_cli = Off
apc.enabled = On
;当一个运行中的服务器中的文件有修改时,如拷贝、解压、编辑时,在该操作完成前,该文件是不完整的,会有一小会的延迟,此时 APC如果对其进行缓存的时候就是有问题的。这个选项就是为了这个延迟时间而设的,默认是2秒,一般无需修改。
apc.file_update_protection = 2
;以下文件将不会被缓存,使用逗号隔开,支持 POSIX 标准的正则表达式
apc.filters =
apc.gc_ttl = 3600
;优化 include_once 和 require_once
apc.include_once_override = Off
apc.max_file_size = 1M
;将会缓存的源文件数,只是一个大概的数字,如果不确定的话可以设置为 0 或者忽略它
apc.num_files_hint = 1000
;优化级别,默认是0,值越高表示优化的级别越高,此参数还在试验中(不过应该有些效果吧,如果不稳定的话就关了它)
apc.optimization = Off
apc.report_autofilter = Off
apc.shm_segments = 1
apc.shm_size = 30
apc.slam_defense = 0
apc.stat = On
;缓存条目在槽中允许的空闲时间(秒),当超过这个时间时,该槽会被其它的缓存条目使用
apc.ttl = 0
apc.user_entries_hint = 100
;同上,只不过是针对用户自定义的缓存条目
apc.user_ttl = 0
;当一个很繁忙的服务器重启或同一时间段内有很多文件同时需要编译时,服务器可能会挂了(或假死:P),当把该项设置为1 时,APC就会每次只编译一个文件,其它没有编译的正在运行中的文件还是以原样运行,完了之后再编译下一个
apc.write_lock = On
下面是APC的一个简单类:
<?php
/**
* APC moudle
*/
class cacheHelper{
//apc_add — Cache a variable in the data store
//apc_cache_info — Retrieves cached information from APC's data store
//apc_clear_cache — Clears the APC cache
//apc_compile_file — Stores a file in the bytecode cache, bypassing all filters.
//apc_define_constants — Defines a set of constants for retrieval and mass-definition
//apc_delete — Removes a stored variable from the cache
//apc_fetch — Fetch a stored variable from the cache
//apc_load_constants — Loads a set of constants from the cache
//apc_sma_info — Retrieves APC's Shared Memory Allocation information
//apc_store — Cache a variable in the data store
public $prefix;
function __construct(){
if(!function_exists('apc_fetch')){
exit("This application must required APC module.");
}
}
/**
* __set
*
* @param mixed $name
* @param mixed $value
* @access public
* @return void
*/
public function __set($name, $value){
apc_store($this->prefix.$name, $value);
}
/**
* __get
*
* @param mixed $name
* @access public
* @return void
*/
public function __get($name){
return apc_fetch($this->prefix.$name);
}
/**
* __isset
*
* @param mixed $name
* @access public
* @return bool
*/
public function __isset($name){
return apc_fetch($this->prefix.$name) ? true : false;
}
/**