一、Memcached
1、memcached 是什么
Memcached是一款开源、高性能、分布式内存对象缓存系统,可应用各种需要缓存的场景,其主要目的是通过降低对Database的访问来加速web应用程序。它是一个基于内存的“键值对”存储,用于存储数据库调用、API调用或页面引用结果的直接数据,如字符串、对象等。Memcached是以LiveJournal旗下Danga Interactive 公司的Brad Fitzpatric 为首开发的一款软件。现在已成为mixi、hatena、Facebook、Vox、LiveJournal等众多服务中提高Web应用扩展性的重要因素。
许多Web应用都将数据保存到RDBMS(关系型数据库管理系统)中,应用服务器从中读取数据并在浏览器中显示。 但随着数据量的增大、访问的集中,就会出现RDBMS的负担加重、数据库响应恶化、 网站显示延迟等重大影响。这时就该memcached大显身手了。memcached是高性能的分布式内存缓存服务器。 一般的使用目的是,通过缓存数据库查询结果,减少数据库访问次数,以提高动态Web应用的速度、 提高可扩展性。如下图:
2、memcached 特点
Memcached是一款开发工具,它既不是一个代码加速器,也不是数据库中间件。其设计哲学思想主要反映在如下方面:
简单key/value存储:服务器不关心数据本身的意义及结构,只要是可序列化数据即可。存储项由“键、过期时间、可选的标志及数据”四个部分组成;
功能的实现一半依赖于客户端,一半基于服务器端:客户负责发送存储项至服务器端、从服务端获取数据以及无法连接至服务器时采用相应的动作;服务端负责接收、存储数据,并负责数据项的超时过期;
各服务器间彼此无视:不在服务器间进行数据同步;
O(1)的执行效率 。
清理超期数据:默认情况下,Memcached是一个LRU缓存,同时,它按事先预订的时长清理超期数据;但事实上,memcached不会删除任何已缓存数据,只是在其过期之后不再为客户所见;而且,memcached也不会真正按期限清理缓存,而仅是当get命令到达时检查其时长;
3、memcached 基于libevent的事件处理
libevent是个程序库,它将Linux的epoll、BSD类操作系统的kqueue等事件处理功能封装成统一的接口。即使对服务器的连接数增加,也能发挥O(1)的性能。memcached使用这个libevent库,因此能在Linux、BSD、Solaris等操作系统上发挥其高性能。
4、memcached 内置内存存储方式
为了提高性能,memcached中保存的数据都存储在memcached内置的内存存储空间中。由于数据仅存在于内存中,因此重启memcached、重启操作系统会导致全部数据消失。另外,内容容量达到指定值之后,就基于LRU(Least Recently Used)算法自动删除不使用的缓存。memcached本身是为缓存而设计的服务器,因此并没有过多考虑数据的永久性问题。
数据存储方式:Slab Allocation
结构图如下:
Slab Allocator的基本原理是按照预先规定的大小,将分配的内存分割成特定长度的块(chunk),并把尺寸相同的块分成组,以完全解决内存碎片问题。但由于分配的是特定长度的内存,因此无法有效利用分配的内存。比如将100字节的数据缓存到128字节的chunk中,剩余的28字节就浪费了。
Slab Allocator 相关术语
Page:分配给Slab的内存空间,默认是1MB。分配给Slab之后根据slab的大小切分成chunk。
Chunk:用于缓存记录的内存空间。
Slab Class:特定大小的chunk的组。
memcached根据收到的数据的大小,选择最适合数据大小的slab。
memcached中保存着slab内空闲chunk的列表,根据该列表选择chunk,然后将数据缓存于其中。
在Slab中缓存记录的原理
下面说明memcached如何针对客户端发送的数据选择slab并缓存到chunk中。memcached根据收到的数据的大小,选择最适合数据大小的slab。 memcached中保存着slab内空闲chunk的列表,根据该列表选择chunk, 然后将数据缓存于其中。如下图,
实际上,Slab Allocator也是有利也有弊。下面介绍一下它的缺点。
Slab Allocator的缺点
Slab Allocator解决了当初的内存碎片问题,但新的机制也给memcached带来了新的问题。这个问题就是,由于分配的是特定长度的内存,因此无法有效利用分配的内存。 例如,将100字节的数据缓存到128字节的chunk中,剩余的28字节就浪费了。如下图,
对于该问题目前还没有完美的解决方案,但在文档中记载了比较有效的解决方案。就是说,如果预先知道客户端发送的数据的公用大小,或者仅缓存大小相同的数据的情况下, 只要使用适合数据大小的组的列表,就可以减少浪费。但是很遗憾,现在还不能进行任何调优,只能期待以后的版本了。 但是,我们可以调节slab class的大小的差别。 接下来说明growth factor(增长系数)选项。
使用Growth Factor进行调优
memcached在启动时指定 Growth Factor因子(通过 -f 选项), 就可以在某种程度上控制slab之间的差异。默认值为1.25。 但是,在该选项出现之前,这个因子曾经固定为2,称为“powers of 2”策略。
让我们用以前的设置,以verbose模式启动memcached试试看:
[root@memcache ~]# memcached -f 2 -vv下面是启动后的verbose输出:slab class 1: chunk size 128 perslab 8192slab class 2: chunk size 256 perslab 4096slab class 3: chunk size 512 perslab 2048slab class 4: chunk size 1024 perslab 1024slab class 5: chunk size 2048 perslab 512slab class 6: chunk size 4096 perslab 256slab class 7: chunk size 8192 perslab 128slab class 8: chunk size 16384 perslab 64slab class 9: chunk size 32768 perslab 32slab class 10: chunk size 65536 perslab 16slab class 11: chunk size 131072 perslab 8slab class 12: chunk size 262144 perslab 4slab class 13: chunk size 524288 perslab 2
可见,从128字节的组开始,组的大小依次增大为原来的2倍。 这样设置的问题是,slab之间的差别比较大,有些情况下就相当浪费内存。 因此,为尽量减少内存浪费,追加了growth factor这个选项。来看看现在的默认设置(f=1.25)时的输出:
slab class 1: chunk size 88 perslab 11915slab class 2: chunk size 112 perslab 9362slab class 3: chunk size 144 perslab 7281slab class 4: chunk size 184 perslab 5698slab class 5: chunk size 232 perslab 4519slab class 6: chunk size 296 perslab 3542slab class 7: chunk size 376 perslab 2788slab class 8: chunk size 472 perslab 2221slab class 9: chunk size 592 perslab 1771slab class 10: chunk size 744 perslab 1409
可见,组间差距比因子为2时小得多,更适合缓存几百字节的记录。 从上面的输出结果来看,可能会觉得有些计算误差, 这些误差是为了保持字节数的对齐而故意设置的。将memcached引入产品,或是直接使用默认值进行部署时, 最好是重新计算一下数据的预期平均长度,调整growth factor, 以获得最恰当的设置。内存是珍贵的资源,浪费就太可惜了。
5、memcached 过期方式
数据过期方式:Lazy Expiration + LRU
Lazy Expirationmemcached内部不会监视记录是否过期,而是在get时查看记录的时间戳,检查记录是否过期。这种技术被称为lazy(惰性)expiration。因此,memcached不会在过期监视上耗费CPU时间。
LRUmemcached会优先使用已超时的记录的空间,但即使如此,也会发生追加新记录时空间不足的情况,此时就要使用名为 Least Recently Used(LRU)机制来分配空间。当memcached的内存空间不足时(无法从slab class 获取到新的空间时),就从最近未被使用的记录中搜索,并将其空间分配给新的记录。
6、memcached 不互相通信的分布式
memcached尽管是“分布式”缓存服务器,但服务器端并没有分布式功能。 各个memcached不会互相通信以共享信息。那么,怎样进行分布式呢? 这完全取决于客户端的实现。如下图,
7、memcached 支持的平台
Linux
FreeBSD
Solaris (memcached 1.2.5以上版本)
Mac OS X
另外也能安装在Windows上。
二、memcached 安装与配置
1、安装libevent
[root@hpf-linux ~]# tar -xf libevent-2.0.22-stable.tar.gz [root@hpf-linux ~]# cd libevent-2.0.22-stable[root@hpf-linux libevent-2.0.22-stable]# ./configure --prefix=/usr/local/libevent[root@hpf-linux libevent-2.0.22-stable]# make && make install
2、安装memcached
[root@hpf-linux ~]# tar -xf memcached-1.4.24.tar.gz [root@hpf-linux ~]# cd memcached-1.4.24[root@hpf-linux memcached-1.4.24]# ./configure --prefix=/usr/local/memcached --with-libevent=/usr/local/libevent[root@hpf-linux memcached-1.4.24]# make && make install
3、配置环境变量
[root@hpf-linux memcached-1.4.24]# vim /etc/profile.d/memcached.shexport PATH=$PATH:/usr/local/memcached/bin/ [root@hpf-linux memcached-1.4.24]# source /etc/profile.d/memcached.sh
4、memcached 启动选项
-p TCP监听端口 (default: 11211)
-U UDP 监听端口 (default: 11211, 0 is off)
-s UNIX socket监听路径,不支持网络
-a UNIX socket访问掩码, 八进制 (default: 0700)
-l 监听的服务器IP地址 (default: all addresses)
-d 启动一个守护进程
-r 最大限度利用核心文件限制
-u 运行memcached用户
-m 最大的内存使用 (default: 64 MB)
-M 内存耗尽返回错误
-c 最大并发连接 (default: 1024)
-k 锁定所有分页内存
-v 输出警告和错误信息
-vv 同时打印客户端请求和返回信息
-vvv 打印内部状态转换信息
-i 打印memcached 和 libevent 版本信息
-P 设置保存pid文件, only used with -d option
-f 块大小增长倍数 (default: 1.25)
-n key+value+flags最小分配空间(default: 48)
-L 如何有效,尝试使用大内存页。增加内存页大小可以减少失误的TLB数量,提高性能。
-D 指定key和IDs的分隔符 default is “:” (colon). 如果指定此选项,统计信息收集自动开启;
-t 使用的线程数量 (default: 4)
-R 每个事件的最大请求数 (default: 20)
-C 禁止使用 CAS
-b 设置积压队列数限制 (default: 1024)、
-B 绑定协议 – one of ascii, binary, or auto (default)
-I 分配给每个slab页(default: 1mb, min: 1k, max: 128m)
-o 配置额外选项
5、启动memcached
[root@hpf-linux ~]# useradd -r -u 490 -s /sbin/nologin memcached[root@hpf-linux ~]# memcached -d -m 64 -u memcached -l 192.168.1.6 -c 256 -P /tmp/memcached.pid -vvv
6、查看监听的端口
[root@hpf-linux ~]# ss -tunlp |grep memcachedudp UNCONN 0 0 192.168.1.6:11211 *:* users:(("memcached",18379,27),("memcached",18379,28),("memcached",18379,29),("memcached",18379,30))tcp LISTEN 0 128 192.168.1.6:11211 *:* users:(("memcached",18379,26))
7、配置SysV脚本
提供配置文件:
[root@hpf-linux ~]# vim /etc/sysconfig/memcachedPORT="11211"USER="memcached"MAXCONN="1024" CACHESIZE="128" OPTIONS=""
[root@hpf-linux ~]# vim /etc/init.d/memcached#!/bin/bash # # Init file for memcached # # chkconfig: - 86 14 # description: Distributed memory caching daemon # # processname: memcached # config: /etc/sysconfig/memcached. /etc/rc.d/init.d/functions## Default variablesPORT="11211" USER="memcached" MAXCONN="1024" CACHESIZE="64" OPTIONS=""[ -f /etc/sysconfig/memcached ] && . /etc/sysconfig/memcachedRETVAL=0prog="/usr/local/memcached/bin/memcached" desc="Distributed memory caching" lockfile="/var/lock/subsys/memcached"start() { echo -n $"Starting $desc (memcached): " daemon $prog -d -p $PORT -u $USER -c $MAXCONN -m $CACHESIZE $OPTIONS RETVAL=$? echo [ $RETVAL -eq 0 ] && touch $lockfile return $RETVAL }stop() { echo -n $"Shutting down $desc (memcached): " killproc $prog RETVAL=$? echo [ $RETVAL -eq 0 ] && rm -f $lockfile return $RETVAL }restart() { stop start }reload() { echo -n $"Reloading $desc ($prog): " killproc $prog -HUP RETVAL=$? echo return $RETVAL }case "$1" in start) start ;; stop) stop ;; restart) restart ;; condrestart) [ -e $lockfile ] && restart RETVAL=$? ;; reload) reload ;; status) status $prog RETVAL=$? ;; *) echo $"Usage: $0 {start|stop|restart|condrestart|status}" RETVAL=1 esac[root@hpf-linux ~]# chmod +x /etc/init.d/memcached[root@hpf-linux ~]# chkconfig --add memcached[root@hpf-linux ~]# chkconfig memcached on
8、启动服务
[root@hpf-linux ~]# service memcached startStarting Distributed memory caching (memcached): [确定]
9、测试
[root@hpf-linux ~]# telnet 192.168.1.6 11211Trying 192.168.1.6...Connected to 192.168.1.6.Escape character is '^]'.statsSTAT pid 8279 #进程IDSTAT uptime 8000 #服务器运行秒数STAT time 1378284623 #服务器当前unix时间戳STAT version 1.4.15 #服务器版本STAT libevent 2.0.21-stable #libevent版本号STAT pointer_size 64 #操作系统指针大小(这台服务器是64位的)STAT rusage_user 0.000999 #进程累计用户时间STAT rusage_system 0.003999 #进程累计系统时间STAT curr_connections 10 #当前打开连接数STAT total_connections 11 #曾打开的连接总数STAT connection_structures 11 #服务器分配的连接结构数STAT reserved_fds 20 #内部使用的FD数STAT cmd_get 0 #执行get命令总数STAT cmd_set 0 #执行set命令总数STAT cmd_flush 0 #执行flush命令总数STAT cmd_touch 0 #执行touch命令总数STAT get_hits 0 #get命中次数STAT get_misses 0 #get未命中次数STAT delete_misses 0 #delete未命中次数STAT delete_hits 0 #delete命中次数STAT incr_misses 0 #incr未命中次数STAT incr_hits 0 #incr命中次数STAT decr_misses 0 #decr未命中次数STAT decr_hits 0 #decr命中次数STAT cas_misses 0 #cas未命中次数STAT cas_hits 0 #cas命中次数STAT cas_badval 0 #使用擦拭次数STAT touch_hits 0 #touch命中次数STAT touch_misses 0 #touch未命中次数STAT auth_cmds 0 #认证处理的次数 STAT auth_errors 0 #认证失败次数STAT bytes_read 7 #读取字节总数STAT bytes_written 0 #写入字节总数STAT limit_maxbytes 134217728 #现在的内存大小为128M STAT accepting_conns 1 #目前接受的新接数STAT listen_disabled_num 0 #失效的监听数STAT threads 4 #当前线程数STAT conn_yields 0 #连接操作主支放弃数目STAT hash_power_level 16 #hash等级STAT hash_bytes 524288 #当前hash表等级STAT hash_is_expanding 0 #hash表扩展大小STAT bytes 0 #当前存储占用的字节数STAT curr_items 0 #当前存储数据总数STAT total_items 0 #启动以来存储的数据总数STAT expired_unfetched 0 #已过期但未获取的对象数目STAT evicted_unfetched 0 #已驱逐但未获取的对象数目STAT evictions 0 #LRU释放的对象数目STAT reclaimed 0 #用已过期的数据条目来存储新数据的数目END
三、memcached 使用详解
1、memcached 基本命令
Memcached提供了为数不多的几个命令来完成与服务器端的交互,这些命令基于memcached的协议实现。
存储类命令:set, add, replace, append, prepend
获取数据类命令:get, delete, incr/decr
统计类命令:stats, stats items, stats slabs, stats sizes
清理命令: flush_all
2、memcached 存储命令的格式
格式:
<command name> <key> <flags> <exptime> <bytes><data block>
参数说明:
<command name> 操作命令:set/add/replace……
<key> 缓存的键值
<flags> 客户机使用它存储关于键值对的额外信息
<exptime> 缓存过期时间 单位为秒 0 表示永远存储
<bytes> 缓存值的字节数
<data block> 数据块