PHP中memcached的使用

1、下载并安装Memcache服务器端

官网,libevent:http://www.monkey.org/~provos/libevent/,Memcache:http://memcached.org/

//1.先安装libevent。这个东西在配置时需要指定一个安装路径,即./configure –prefix=/usr
//2.再安装memcached,只是需要在配置时需要指定libevent的安装路径即./configure –with-libevent=/usr

cd /usr/local/src
wget https://sourceforge.net/projects/levent/files/libevent/libevent-2.0/libevent-2.0.22-stable.tar.gz
wget http://www.memcached.org/files/memcached-1.4.24.tar.gz
wget https://pecl.php.net/get/memcached-2.2.0.tgz
wget https://launchpad.net/libmemcached/1.0/1.0.18/+download/libmemcached-1.0.18.tar.gz

tar zxf libmemcached-1.0.18.tar.gz
cd libmemcached-1.0.18
./configure --prefix=/usr/local/libmemcached --with-memcached
make && make install

tar zxf libevent-2.0.22-stable.tar.gz 
cd libevent-2.0.22-stable
 ./configure  --prefix=/usr
make && make install

cd /usr/local/src
tar zxf memcached-1.4.24.tar.gz
cd memcached-1.4.24
./configure --with-libevent=/usr
make && make install
/usr/local/bin/memcached -d -m 10 -u root -l 192.168.1.30 -p 12000 -c 256 -P /tmp/memcached.pid

cd /usr/local/src
tar zxf memcached-2.2.0.tgz
cd memcached-2.2.0
phpize
./configure --enable-memcached --with-php-config=php-config --with-libmemcached-dir=/usr/local/libmemcached --disable-memcached-sasl
make && make install
echo 'extension=memcached.so' > /etc/php.d/memcached.ini
service php-fpm reload

参数解释

-d选项是启动一个守护进程
-m是分配给Memcache使用的内存数量,单位是MB,我这里是10MB
-u是运行Memcache的用户,我这里是root
-l是监听的服务器IP地址,如果有多个地址的话,我这里指定了服务器的IP地址192.168.1.30
-p是设置Memcache监听的端口,我这里设置了12000,最好是1024以上的端口
-c选项是最大运行的并发连接数,默认是1024,我这里设置了256,按照你服务器的负载量来设定
-P是设置保存Memcache的pid文件,我这里是保存在 /tmp/memcached.pid

防火墙

# iptables -A INPUT -p tcp -s 192.168.1.30 –dport 12000 -j ACCEPT
# iptables -A INPUT -p udp -s 192.168.1.30 –dport 12000 -j ACCEPT

想开机自动启动的话,只需在/etc/rc.d/rc.local中加入一行命令就可以了

2、测试

<?php
$m = new Memcached();

/* Add 2 servers, so that the second one
   is twice as likely to be selected. */
$m->addServer('192.168.1.30', 12000, 33);

$v=str_repeat('a',1024*1024);

if( $m->add('mystr',$v,3600)){
        echo  '原始数据缓存成功!';
}else{
    echo '数据已存在:'.$m->get("mystr");
}

memcached set数据的时候是默认压缩的,可以通过setOption(memcahed::OPT_COMPRESSION,0)来设置,也就是说memcached server不能存储超过1M的数据但是经过客户端压缩数据后,只要小于1M的数据都能存储成功

3、memcached的操作

Memcached — Memcached类

    Memcached::add — 向一个新的key下面增加一个元素
    Memcached::addByKey — 在指定服务器上的一个新的key下增加一个元素
    Memcached::addServer — 向服务器池中增加一个服务器
    Memcached::addServers — 向服务器池中增加多台服务器
    Memcached::append — 向已存在元素后追加数据
    Memcached::appendByKey — 向指定服务器上已存在元素后追加数据
    Memcached::cas — 比较并交换值
    Memcached::casByKey — 在指定服务器上比较并交换值
    Memcached::__construct — 创建一个Memcached实例
    Memcached::decrement — 减小数值元素的值
    Memcached::decrementByKey — Decrement numeric item's value, stored on a specific server
    Memcached::delete — 删除一个元素
    Memcached::deleteByKey — 从指定的服务器删除一个元素
    Memcached::deleteMulti — Delete multiple items
    Memcached::deleteMultiByKey — Delete multiple items from a specific server
    Memcached::fetch — 抓取下一个结果
    Memcached::fetchAll — 抓取所有剩余的结果
    Memcached::flush — 作废缓存中的所有元素
    Memcached::get — 检索一个元素
    Memcached::getAllKeys — Gets the keys stored on all the servers
    Memcached::getByKey — 从特定的服务器检索元素
    Memcached::getDelayed — 请求多个元素
    Memcached::getDelayedByKey — 从指定的服务器上请求多个元素
    Memcached::getMulti — 检索多个元素
    Memcached::getMultiByKey — 从特定服务器检索多个元素
    Memcached::getOption — 获取Memcached的选项值
    Memcached::getResultCode — 返回最后一次操作的结果代码
    Memcached::getResultMessage — 返回最后一次操作的结果描述消息
    Memcached::getServerByKey — 获取一个key所映射的服务器信息
    Memcached::getServerList — 获取服务器池中的服务器列表
    Memcached::getStats — 获取服务器池的统计信息
    Memcached::getVersion — 获取服务器池中所有服务器的版本信息
    Memcached::increment — 增加数值元素的值
    Memcached::incrementByKey — Increment numeric item's value, stored on a specific server
    Memcached::isPersistent — Check if a persitent connection to memcache is being used
    Memcached::isPristine — Check if the instance was recently created
    Memcached::prepend — 向一个已存在的元素前面追加数据
    Memcached::prependByKey — Prepend data to an existing item on a specific server
    Memcached::quit — 关闭所有打开的链接。
    Memcached::replace — 替换已存在key下的元素
    Memcached::replaceByKey — Replace the item under an existing key on a specific server
    Memcached::resetServerList — Clears all servers from the server list
    Memcached::set — 存储一个元素
    Memcached::setByKey — Store an item on a specific server
    Memcached::setMulti — 存储多个元素
    Memcached::setMultiByKey — Store multiple items on a specific server
    Memcached::setOption — 设置一个memcached选项
    Memcached::setOptions — Set Memcached options
    Memcached::setSaslAuthData — Set the credentials to use for authentication
    Memcached::touch — Set a new expiration on an item
    Memcached::touchByKey — Set a new expiration on an item on a specific server

 参见:http://php.net/manual/zh/book.memcached.php

注意事项:key的长度不能大于250字符,缓存对象的大小不能大于1MB,item对象的过期时间最长可以达到30天

如果使用的Memcached客户端支持"key的前缀"或类似特性,那么key(前缀+原始key)的最大长度是可以超过250个字符的。

推荐使用较短的key,这样可以节省内存和带宽。

4、适用的场景

1)如果网站包含了访问量很大的动态网页,因而数据库的负载将会很高。由于大部分数据库请求都是读操作,那么memcached可以显著地减小数据库负载

2)如果数据库服务器的负载比较低但CPU使用率很高,这时可以缓存计算好的结果( computed objects )和渲染后的网页模板(enderred templates)

3)利用memcached可以缓存session数据、临时数据以减少对他们的数据库写操作

4)缓存一些很小但是被频繁访问的文件

5)缓存Web 'services'(非IBM宣扬的Web Services,译者注)或RSS feeds的结果

 

5、memcached的原理

1)Big-O

memcache的大部分功能(add,get,set,flush等)都是O(1)的操作,这意味着它们是恒定的时间功能,无论缓存里存了多少东西,这功能将只需要获得缓存中的一项,所以你不能遍历所有项目里的item

2)LRU算法

LRU是Least Recently Used的缩写,即最近最少使用页面置换算法,是为虚拟页式存储管理服务的,它将删除最长时间不使用的数据。该数据可能并不是最大的数据,也不是第一个存储在缓存中的数据。

在实现时,一般使用一个哈希表类存储被缓存的对象,和一个双向链表类存储对象被使用的情况

所有的对象有有一个counter,这个counter记录了一个时间戳。当一个新的对象被创建的时候,counter就会被设置成当前时间。当一个对象被读取的时候,就会重置counter为当前的时间。一旦memcache需要为一个新的对象腾出空间而剔除旧对象时,只需要找到最低的counter对应的那个对象,这个对象要么是一直没有被读取或者是很长时间之前被读取(说明这个对象不需要了,否这的话该对象的counter应该会很接近当前时间)。

这样创建了一个简单系统,并使用了非常高效的缓存。如果它不被使用,就会被踢出系统。

3)slab allocation

Memcached的内存分配以Page为单位,Page默认值为1M,可以在启动时通过-I参数来指定。

Slab Allocation的原理——将分配的内存分割成各种尺寸的块(chunk), 并把尺寸相同的块分成组(chunk的集合),每个chunk集合被称为slab

Slab是由多个Page组成的,Page按照指定大小切割成多个chunk

Growth Factor

memcached在启动时通过-f选项可以指定 Growth Factor因子。该值控制slab之间的差异,chunk大小的差异。默认值为1.25。

通过memcached-tool查看指定memcached实例的不同slab状态,可以看到各Item所占大小(chunk大小)差距为1.25

[root@localhost src]# memcached -u root  -vv
slab class   1: chunk size        80 perslab   13107
slab class   2: chunk size       104 perslab   10082
slab class   3: chunk size       136 perslab    7710
slab class   4: chunk size       176 perslab    5957
slab class   5: chunk size       224 perslab    4681
slab class   6: chunk size       280 perslab    3744
slab class   7: chunk size       352 perslab    2978
slab class   8: chunk size       440 perslab    2383
slab class   9: chunk size       552 perslab    1899
slab class  10: chunk size       696 perslab    1506
slab class  11: chunk size       872 perslab    1202
slab class  12: chunk size      1096 perslab     956
slab class  13: chunk size      1376 perslab     762
slab class  14: chunk size      1720 perslab     609
slab class  15: chunk size      2152 perslab     487
slab class  16: chunk size      2696 perslab     388
slab class  17: chunk size      3376 perslab     310
slab class  18: chunk size      4224 perslab     248
slab class  19: chunk size      5280 perslab     198
slab class  20: chunk size      6600 perslab     158
slab class  21: chunk size      8256 perslab     127
slab class  22: chunk size     10320 perslab     101
slab class  23: chunk size     12904 perslab      81
slab class  24: chunk size     16136 perslab      64
slab class  25: chunk size     20176 perslab      51
slab class  26: chunk size     25224 perslab      41
slab class  27: chunk size     31536 perslab      33
slab class  28: chunk size     39424 perslab      26
slab class  29: chunk size     49280 perslab      21
slab class  30: chunk size     61600 perslab      17
slab class  31: chunk size     77000 perslab      13
slab class  32: chunk size     96256 perslab      10
slab class  33: chunk size    120320 perslab       8
slab class  34: chunk size    150400 perslab       6
slab class  35: chunk size    188000 perslab       5
slab class  36: chunk size    235000 perslab       4
slab class  37: chunk size    293752 perslab       3
slab class  38: chunk size    367192 perslab       2
slab class  39: chunk size    458992 perslab       2
slab class  40: chunk size    573744 perslab       1
slab class  41: chunk size    717184 perslab       1
slab class  42: chunk size   1048576 perslab       1
<26 server listening (auto-negotiate)
<27 server listening (auto-negotiate)
<28 send buffer was 112640, now 268435456
<32 send buffer was 112640, now 268435456
<28 server listening (udp)
<31 server listening (udp)
<32 server listening (udp)
<35 server listening (udp)
<30 server listening (udp)
<29 server listening (udp)
<34 server listening (udp)
<33 server listening (udp)

Slab Allocation的缺点

Slab Allocation可以有效的解决内存碎片问题,但是在如下情况下,会导致内存的浪费:

每个slab的chunk大小是固定的,当item的占用空间实际小于chunk大小时,会出现内存浪费

每个slab的大小是固定的(因为page是固定的),当slab不能被他所拥有的chunk整除时,会出现内存浪费

按照Growth Factor因子生成指定大小的slab,而某slab id根本未被使用时,会出现内存浪费

 

 4)分布式、一致性hash算法

转载于:https://www.cnblogs.com/chenpingzhao/p/4681666.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值