安装及配置APC

时间:14-06-17 栏目:软件架构技术 作者:乐测网 评论:0 点击: 585 次

APC主要有两个作用,一是将php的编译缓存保存在共享内存中,简单理解就是使用了APC以后,省去了需要每次将PHP源代码编译为PHP Opcode的时间,提高了性能,只要下次PHP源代码没有任何修改,则就不需要重新编译了。另一方面的作用是类似Memcached的功能,APC是将数据保存到共享内存中,Memcached也是将数据保存在内存中,不过Memcached支持分布式,而APC不支持分布式。

1206683225175433134

1.安装APC
http://pecl.php.net/package/apc找到最新的apc稳定版本
# wget http://pecl.php.net/get/APC-3.0.19.tgz
--17:47:42--  http://pecl.php.net/get/APC-3.0.19.tgz
Resolving pecl.php.net... 216.92.131.66
Connecting to pecl.php.net|216.92.131.66|:80... connected.
HTTP request sent, awaiting response... 200 OK
Length: 115735 (113K) [application/octet-stream]
Saving to: `APC-3.0.19.tgz'

100%[=====================================================================================>] 115,735     50.1K/s   in 2.3s

17:47:48 (50.1 KB/s) - `APC-3.0.19.tgz' saved [115735/115735]

# tar xzvf APC-3.0.19.tgz
# cd APC-3.0.19
# /usr/local/webserver/php/bin/phpize
Configuring for:
PHP Api Version:         20041225
Zend Module Api No:      20060613
Zend Extension Api No:   220060519
./configure --enable-apc-mmap --enable-filehits --enable-apc
make
make install

**********************************************
--enable-filehits主要用于函数apc_cache_info中,说明如下
array apc_cache_info ([ string $cache_type [, bool $limited=false ]] )

cache_type
If cache_type is "user", information about the user cache will be returned.

If cache_type is "filehits", information about which files have been served from the bytecode cache for the current request will be returned. This feature must be enabled at compile time using --enable-filehits.

If an invalid or no cache_type is specified, information about the system cache (cached files) will be returned.
**********************************************

# make install
Installing shared extensions:     /usr/local/webserver/php/lib/php/extensions/no-debug-non-zts-20060613/

apc.so文件已经生成
# ll /usr/local/webserver/php/lib/php/extensions/no-debug-non-zts-20060613/
total 4072
-rwxr-xr-x 1 root root  416983 Sep 24 18:00 apc.so
-rwxr-xr-x 1 root root  172209 May 17 19:07 db4.so
-rwxr-xr-x 1 root root  517843 Apr 28 20:55 eaccelerator.so
-rwxr-xr-x 1 root root  748762 Jul 22 03:30 imagick.so
-rwxr-xr-x 1 root root  185627 Apr 28 20:38 memcache.so
-rwxr-xr-x 1 root root  118580 Apr 28 22:23 pdo_mysql.so
-rwxr-xr-x 1 root root 1308827 Jul  8 19:10 xapian.so
-rwxr-xr-x 1 root root  620251 Aug  5 03:41 xdebug.so

 

2.APC的相关配置
相关配置
---------------------------------
vi /usr/local/webserver/php/etc/php.ini,加入如下的一行
extension = "apc.so"
然后执行php -m | grep apc,发现已经出现apc模块了,证明apc已经生效了
# php -m | grep apc
apc

上面只是命令行生效了,为了让web服务器也生效,必须要重启php-fpm(我的环境是Nginx+fastcgi方式执行的php)
# /usr/local/webserver/php/sbin/php-fpm restart
Shutting down php_fpm . done
Starting php_fpm  done

vi index.php,输入如下的内容
<?php
phpinfo()
?>
在浏览器执行index.php即可看到模块apc已经生效
下面的输出是在php.ini中没有加任何apc的配置时的输出值

apc
1206683225175433135

Directive Local Value Master Value
1206683225175433136

另外apc可以在php.ini文件中进行相关的配置,详细的配置可以看

或者可以查看php官方的英文配置说明http://cn.php.net/manual/en/apc.configuration.php

vi /usr/local/webserver/php/etc/php.ini,加入如下的几行
[APC]
apc.cache_by_default = on
apc.enabled = on
apc.max_file_size = 1m
apc.shm_segments = 3
apc.shm_size = 50
apc.stat = on
apc.write_lock = On
然后重启php-fpm进程

在php.ini中已经修改的配置在phpinfo页面中已经生效了
apc.shm_segments 3 3
apc.shm_size 50 50

同时发现了一个问题,发现apc与eAccelerator冲突,如果在php中同时安装了apc与eAccelerator,则apc的缓存文件的功能不会生效(或者你安装了其它的php加速器如xCache,Zend Optimizer,如果apc的系统缓存没有生效,则试着禁止这些看看效果)
;[eaccelerator]
;zend_extension="/usr/local/webserver/php/lib/php/extensions/no-debug-non-zts-20060613/eaccelerator.so"
;eaccelerator.shm_size="128"
;eaccelerator.cache_dir="/usr/local/webserver/eaccelerator_cache"
;eaccelerator.enable="1"
;eaccelerator.optimizer="1"
;eaccelerator.check_mtime="1"
;eaccelerator.debug="0"
;eaccelerator.filter=""
;eaccelerator.shm_max="0"
;eaccelerator.shm_ttl="300"
;eaccelerator.shm_prune_period="120"
;eaccelerator.shm_only="0"
;eaccelerator.compress="1"
;eaccelerator.compress_level="9"

 

3.测试APC缓存用户数据的功能(类似于Memcached)
在apc的安装目录里面有一个apc.php文件,拷贝它到Web服务器所在目录下,这个程序可以监控apc的各种状态数据
# cd /home/software/APC-3.0.19
# cp apc.php /home/htdocs/www/

vi apc1.php,输入如下的内容
<?php
define("LINE", "<br />\n");

$apc_cached = ini_get('apc.cache_by_default');
echo "aa:$apc_cached";
echo $apc_cached ? 'apc cache file' : 'apc not cache file';
echo LINE;
function microtime_float()
{
list($usec, $sec) = explode(" ", microtime());
return ((float)$usec + (float)$sec);
}

$time_start = microtime_float();

$key = 'name';
$value = 'caihuafeng';

//调用apc_add时对应的key已经存在,则不会覆盖,函数的返回值是false
$ret = apc_add($key, $value);
echo "ret:$ret<br />\n";
$ret2 = apc_add($key, $value);
echo "ret2:$ret2<br />\n";

//调用apc_store时如果对应的key已经存在,则直接覆盖
$ret3 = apc_store($key, $value);
echo "ret3:$ret3<br />\n";
for ($i = 0; $i < 100; $i++) {
apc_store($i, $i);
}

//删除指定的键对应的内容
//apc_delete($key);

$cache_type = $_REQUEST['cache_type'] ? $_REQUEST['cache_type'] : 'user';
echo "cache_type:", $cache_type, LINE;
/*
array apc_cache_info ([ string $cache_type [, bool $limited=false ]] )
Retrieves cached information and meta-data from APC's data store.

cache_type
If cache_type is "user", information about the user cache will be returned.

If cache_type is "filehits", information about which files have been served from the bytecode cache for the current request will be returned. This feature must be enabled at compile time using --enable-filehits.

If an invalid or no cache_type is specified, information about the system cache (cached files) will be returned.
*/
echo 'apc_cache_info';
var_dump(apc_cache_info($cache_type));

/*
bool apc_clear_cache ([ string $cache_type ] )
Clears the user/system cache.

cache_type
If cache_type is "user", the user cache will be cleared; otherwise, the system cache (cached files) will be cleared.
*/
//apc_clear_cache($cache_type);

echo 'name:' . apc_fetch('name') . "<br />\n";
echo "<hr />\n";

$constants = array(
'ONE'   => 1,
'TWO'   => 2,
'THREE' => 3,
);
apc_define_constants('numbers', $constants);
echo 'one, two, three:', ONE, TWO, THREE, LINE;

$constants = array(
'ONE'   => 1,
'TWO'   => 2,
'THREE' => 3,
);
apc_define_constants('numbers', $constants);
apc_load_constants('numbers');
echo 'one, two, three:', ONE, TWO, THREE, LINE;

/*
array apc_sma_info ([ bool $limited=false ] )
Retrieves APC's Shared Memory Allocation information.

limited
When set to FALSE (default) apc_sma_info() will return a detailed information about each segment.
*/
echo 'apc_sam_info';
var_dump(apc_sma_info(false));

echo 'Request time:', $_SERVER['REQUEST_TIME'], LINE;

$time_end = microtime_float();
$time = $time_end - $time_start;

echo "Spend time in $time seconds", LINE;
?>
输出结果就不在此写出来了

vi apc2.php,输入如下的内容
<?php
define("LINE", "<br />\n");
echo 'name:' . apc_fetch('name') . "<br />\n";
?>
输出:name:caihuafeng
我们看到apc2.php文件中只有apc_fetch,而没有apc_add或apc_store的操作,说明数据已经保存到共享内存中去了

lsof | grep apc时有如下的输出,不止下面这些;通过浏览器执行完程序你会发现,/tmp/apc*这些文件根本上是不存在的,可能是进程执行完了以后就立即删除了,DEL应该就是表示删除的意思
php-cgi   29349       www  DEL       REG        3,1               32854 /tmp/apc.khQ6VN
php-cgi   29350       www  mem       REG        3,1   416983     224539 /usr/local/webserver/php/lib/php/extensions/no-debug-n
on-zts-20060613/apc.so
php-cgi   29350       www  DEL       REG        3,1               32854 /tmp/apc.khQ6VN
php-cgi   29351       www  mem       REG        3,1   416983     224539 /usr/local/webserver/php/lib/php/extensions/no-debug-n
on-zts-20060613/apc.so
php-cgi   29351       www  DEL       REG        3,1               32854 /tmp/apc.khQ6VN
php-cgi   29352       www  mem       REG        3,1   416983     224539 /usr/local/webserver/php/lib/php/extensions/no-debug-n
on-zts-20060613/apc.so

# ls /tmp/
gconfd-root     keyring-lJxrr7  keyring-vcAy69  keyring-Z3N4md  mysql.sock                       VMwareDnD
keyring-9zOctR  keyring-mPfuTA  keyring-vD5JCr  keyring-zcyUaJ  scim-helper-manager-socket-root  vmware-root
keyring-ayJSKz  keyring-T5LSOn  keyring-vYf2v8  mapping-root    scim-panel-socket:0-root         xdebug
keyring-AZF1bf  keyring-T6jLxb  keyring-x3Jeeh  memcached.pid   scim-socket-frontend-root
keyring-fkCrUe  keyring-tMima3  keyring-XhEixk  mysqlslow.log   virtual-root.xJQE1h

# ls /tmp/ | grep apc.*

通过ipcs命令查看共享内存的内容也没有什么变化,这就有点奇怪了,这个问题以后再来研究
# ipcs

------ Shared Memory Segments --------
key        shmid      owner      perms      bytes      nattch     status
0x00000000 32768      gdm       600        393216     2          dest

------ Semaphore Arrays --------
key        semid      owner      perms      nsems

------ Message Queues --------
key        msqid      owner      perms      used-bytes   messages

 

4.测试APC缓存系统数据的功能(也就是缓存文件的编译结果)
a.启动缓存系统数据的功能
[APC]
apc.cache_by_default = On
apc.enabled = On
apc.max_file_size = 1m
apc.shm_segments = 1
apc.shm_size = 64
apc.stat = Off
apc.write_lock = On
apc.mmap_file_mask = /tmp/apc.XXXXXX
apc.ttl = 7200
apc.user_ttl = 7200
apc.gc_ttl=3600
apc.num_filters_hint = 1024

apc1.php在启用apc功能的时候的请求时间
apc enabled
Spend time in 0.0270249843597 seconds
Spend time in 0.0296368598938 seconds
Spend time in 0.0324420928955 seconds
Spend time in 0.0270788669586 seconds
Spend time in 0.0330448150635 seconds

APC编译数据缓存统计,index.php因为是第一次访问,因此Hits为0,其它的文件均不是第一次访问,因此Hits均大于0。

b.禁止缓存系统数据的功能
将apc.cache_by_default的值设为Off,其它的值不变

为了保证测试结果的准确,将php-fpm及Web服务器nginx均重启
# ../sbin/php-fpm restart
Shutting down php_fpm  done
Starting php_fpm  done
# killall nginx
# ../../nginx/sbin/nginx

此时查看apc.php的输出结果中的值,说明此时共享内存中没有缓存任何数据
File Cache Information
Cached Files 0 ( 0.0 Bytes)
Hits 1
Misses 0

User Cache Information
Cached Variables 0 ( 0.0 Bytes)
Hits 0
Misses 0

再次测试apc1.php的程序执行时间,当然也有可能禁止了系统缓存以后程序执行时间比启动系统缓存以后程序执行时间还要短,当然这里面有其它的因素影响,比如恰好是服务器最繁忙的时候,那我们来计算一下平均值,这样可能更加准确一点,看看下面的数据我们就知道APC的系统缓存起作用了,如果php程序的业务逻辑更加复杂,估计效果更加明显。

apc disabled
Spend time in 0.02969789505 seconds
Spend time in 0.127185106277 seconds(比启用系统缓存时5次测试的任何一次都要长)
Spend time in 0.0369350910187 seconds(比启用系统缓存时5次测试的任何一次都要长)
Spend time in 0.0839560031891 seconds(比启用系统缓存时5次测试的任何一次都要长)
Spend time in 0.0326659679413 seconds(比启用系统缓存时5次测试的任何一次都要长)

至少有一点,现在看apc.php中的统计信息也表明系统缓存生效了
File Cache Information
Cached Files 2 (379.5 KBytes)
Hits 52
Misses 2

Cached Variables 102 ( 9.6 KBytes)
Hits 16
Misses 0

 

5.测试系统数据缓存的其它方法
测试apc的系统数据缓存是否生效还有一个方法,那就是修改php的源代码
比如apc1.php中开始的几行原来为
$apc_cached = ini_get('apc.cache_by_default');
echo $apc_cached ? 'apc cache file' : 'apc not cache file';

现改为
$apc_cached = ini_get('apc.cache_by_default');
echo "New added:$apc_cached";
echo $apc_cached ? 'apc cache file' : 'apc not cache file';

按ctrl+f5强制刷新页面,页面的输出中完全没有new added的字样,这就可以说明apc的系统缓存已经生效了(也就是执行的是编译结果的缓存)
apc not cache file
ret:
ret2:
ret3:1
cache_type:user
apc_cache_info
......
......

有的人在这里可能会有疑问了,要是我的php程序修改了,而apc的系统缓存又在生效中,那我要看最新的执行结果怎么办?
有如下的几种方法:
a.用apc_compile_file编译修改过的文件,我们可以参考一下facebook的代码
$path = '/path/to/source';
$exclude = $path.'/www/admin/';
$expr = '.*\\.php';
$files = split(' ', exec("find -L $path -regex '$expr'
| grep -v '$exclude' | xargs"));
// prime php files
foreach($files as $file) {
apc_compile_file($file);
}

b.重启web服务器后所有的用户数据缓存及系统数据缓存都将失效,这样我们通过浏览器访问php程序时当然是最新的php程序返回的结果。

c.将apc.stat设为on,表示apc在每次请求脚本时都检查脚本是否被更新,如果被更新则自动重新编译和缓存编译后的内容。但这样做对性能有不利影响。

d.如果apc.ttl的值不为0(比如3600,表示1小时),则在web服务器不重启的情况下,则只能等待缓存失效了,如果要求比较及时,就只能采用方法a。提示php.ini中与ttl有关的配置只是对系统缓存有效,而用户缓存是通过函数apc_store或apc_add中的第三个参数ttl来控制的。

apc.ttl = 0
;sys
; 缓存条目在缓冲区中允许逗留的秒数。0 表示永不超时。建议值为7200~36000。
; 设为 0 意味着缓冲区有可能被旧的缓存条目填满,从而导致无法缓存新条目。

apc.user_ttl = 0
;sys
; 类似于apc.ttl,只是针对每个用户而言,建议值为7200~36000。
; 设为 0 意味着缓冲区有可能被旧的缓存条目填满,从而导致无法缓存新条目。

apc.gc_ttl = 3600
;sys
; 缓存条目在垃圾回收表中能够存在的秒数。
; 此值提供了一个安全措施,即使一个服务器进程在执行缓存的源文件时崩溃,
; 而且该源文件已经被修改,为旧版本分配的内存也不会被回收,直到达到此ttl值为止。
; 设为零将禁用此特性。

个人认为比较好的方法是a(此时将apc.stat设为off),如果有哪些php文件修改,就重新编译它,如果有多台Web服务器,则通过rsync之类的软件自动同步即可,如果要求比较及时的话,则只能手动通过scp之类的命令同步了,因为rsync多少有些延迟。

声明: 本文由( 乐测网 )原创编译,转载请保留链接: 安装及配置APC

安装及配置APC:等您坐沙发呢!

发表评论(需填写下方的验证回答,无用户名的评论将视为匿名发表,您也可以登陆或注册后再发表评论)

您必须 [ 登录 ] 才能发表留言!

@关注乐测网

-----==== 本站公告 ====-----
1.本站是目前唯一能支持智能手机平板电脑访问的软件测试技术网站.
2.具体访问方法请参考本站的手机访问说明,或直接点击以下链接:

→点击这里打开手机平板访问说明←

    智能手机平板电脑访问方法

1.用手机浏览器输入本站网址http://www.test666.com就可直接访问
2.微信、QQ、微博或手机浏览器的二维码扫描功能扫描以下二维码图

标签云球