redis下載: http://redis.io/download
php
多臺服務安裝: http://www.vquickphp.com/?a=blogview&id=30git
開機加自啓動:echo "redis-server /the_path_to_redis_conf/redis.conf" >>/etc/rc.local github
之前有想過用 Memcache 實現M/S架構的負載均衡方案,直到據說了 Redis 後才發現它作得更好。redis
發了幾天時間研究了一下 Redis ,感受真的很不錯,特整理一下!算法
[memecache 特色]shell
1:速度最快(沒有自測,但網上有詳細的測試用例)數據庫
2:支持水平擴展,能夠任意添加節點緩存
[redis 特色]安全
1:速度沒有memcache快bash
2:支持M/S的主從備份
3:能夠支持多數據庫
4:操做指令很豐富
4:支持異步數據持久化(以文件保存)
總結:
1:若是是簡單的數據緩存建議使用MEMCACHE。
2:若是要對單一操做的數據量很是的大則使用MEMCACHE
3: 若是想作性能很好的緩存集羣能夠用Redis(M/S讀寫分離,如weibo中的排行榜等)
4: 若是在高併發下又想保存數據則能夠用Redis (如更新熱門文章瀏覽次數,memcachedb也不錯)
將 redis 安裝到 /usr/local/redis
1:從 http://redis.io/download 上下載安裝包
#配置文件目錄 mkdir -p /usr/local/redis/conf #pid目錄 mkdir -p /usr/local/redis/run #數據目錄 mkdir -p /usr/local/redis/db wget http://redis.googlecode.com/files/redis-2.2.15.tar.gz tar -zxvf redis-2.2.15.tar.gz cd redis-2.2.15 make #make install 時能夠指定安裝目錄 這樣的話會將可執行文件存放在 PREFIX/bin下面 make PREFIX=/usr/local/redis install cp redis.conf /usr/local/redis/conf/ #若是你沒有在 make install 時指定安裝目錄 則能夠將一下文件 cp 便可 cd src # 將 src 目錄下全部可執行文件複製到安裝目錄 cp redis-benchmark redis-check-aof redis-check-dump redis-cli redis-server mkreleasehdr.sh /usr/local/redis/
2:修改配置文件中的以下選項 其餘的配置項可在本文最後面看到
-------vi /usr/local/redis/conf/redis.conf --------
#以守護進程模式運行 把redis集成到系統服務中時也方便 daemonize yes #指定pid的存放位置方便殺死進程 pidfile /usr/local/redis/run/redis.pid #指定redis數據文件的存放位置,即dump.rdb(數據持久化文件)和appendonly.aof(操做記錄文件)的存放位置 dir /usr/local/redis/db
--------------------------------------------------------------
3:建立服務腳本
------- vi /usr/local/redis/start.sh ---------
#!/bin/bash redis-server /usr/local/redis/conf/redis.conf
--------------------------------------------------------
------- vi /usr/local/redis/stop.sh ---------
#!/bin/bash kill `cat /usr/local/redis/run/redis.pid`
--------------------------------------------------------
chmod a+x /usr/local/redis/start.sh /usr/local/redis/stop.sh
4:啓動 redis 服務
/usr/local/redis/start.sh
驗證證服務是否成功:
netstat -tunlp | grep 6379
5:啓動客戶端驗證
/usr/local/redis/redis-cli >set key1 val1 >get key1
6:或者將redis做爲一個系統服務
cp redis-2.2.15/utils/redis_init_script /etc/init.d/redis
vi /etc/init.d/redis #!/bin/sh #添加chkconfig 配置 #chkconfig: 2345 80 90 # Simple Redis init.d script conceived to work on Linux systems # as it does use of the /proc filesystem. # 會被後續加載的CONF文件覆蓋 REDISPORT=6379 #改成你的安裝路徑 EXEC=/usr/local/redis/bin/redis-server CLIEXEC=/usr/local/redis/bin/redis-cli #這裏指定沒太大意義 會被CONF的配置文件覆蓋 PIDFILE=/var/run/redis_${REDISPORT}.pid #加載的配置文件路徑 CONF="/usr/local/redis/conf/redis.conf"
主要添加 chkconfig配置 設置 EXEC CLIEXEC CONF 的正確路徑便可 REDISPORT PIDFILE 什麼的會被CONF給覆蓋掉
#將服務註冊到系統 就能夠了 chkconfig --add redis #使用命令 service redis start/stop/restart/status
這裏咱們以本機配置 1臺Master + 2臺Slave 爲例子,其中:
Master IP:127.0.0.1 PORT:6379 Slave1 IP:127.0.0.1 PORT:63791 Slave2 IP:127.0.0.1 PORT:63792
1:複製兩個 Slave 目錄,方便管理
cp -r /usr/local/redis /usr/local/redis_slave_1 cp -r /usr/local/redis /usr/local/redis_slave_2
2:修改 redis-slave1 的配置文件和服務腳本
-- vi /usr/local/redis_slave_1/conf/redis.conf --
port 63791 pidfile /usr/local/redis_slave_1/run/redis.pid dir /usr/local/redis_slave_1/db slaveof 127.0.0.1 6379
----------------------------------------------------------
--------vi /usr/local/redis_slave_1/start.sh -------
#!/bin/bash redis-server /usr/local/redis_slave_1/conf/redis.conf
---------------------------------------------------
------- vi /usr/local/redis_slave_1/stop.sh ---------
#!/bin/bash kill `cat /usr/local/redis_slave_1/run/redis.pid`
--------------------------------------------------------
3:修改 redis_slave_2 的配置文件和服務腳本
-- vi /usr/local/redis_slave_2/conf/redis.conf --
port 63792 pidfile /usr/local/redis_slave_2/run/redis.pid dir /usr/local/redis_slave_2/db slaveof 127.0.0.1 6379
----------------------------------------------------------
-------- vi /usr/local/redis_slave_2/start.sh --------
#!/bin/bash redis-server /usr/local/redis_slave_2/conf/redis.conf
---------------------------------------------------
------- vi /usr/local/redis_slave_2/stop.sh ---------
#!/bin/bash kill `cat /usr/local/redis_slave_2/run/redis.pid`
--------------------------------------------------------
4:啓動 M/S 服務
/usr/local/redis/start.sh /usr/local/redis_slave_1/start.sh /usr/local/redis_slave_2/start.sh
驗證服務是否正常: netstat -tunlp | grep redis-server 有3個端口存在證實成功了
0 0.0.0.0:6379 0 0.0.0.0:63791 0 0.0.0.0:63792
5:驗證M/S服務是否生效
redis-cli [默認鏈接端口:6379 的 Master服務] >set key1 val1 >quit
redis-cli -p 63791 [鏈接 slave_1服務] >get key1 "val1" (數據成功同步了)
redis-cli -p 63792 [鏈接 slave_2服務] >get key1 "val1" (數據成功同步了)
Redis全部的客戶端在 http://redis.io/clients [PHP選項卡] 基於性能選擇安裝 phpredis
在 https://github.com/phpredis/phpredis 上下載源代碼包到本地而後上傳到服務器.
tar -zxvf phpredis-2.2.7.tar.gz cd phpredis-2.2.7 /usr/local/php/bin/phpize
若是出現:
Configuring for:
PHP Api Version: 20041225
Zend Module Api No: 20060613
Zend Extension Api No: 220060519
Cannot find autoconf. Please check your autoconf installation and the $PHP_AUTOCONF environment variable is set correctly and then rerun this script.
用下面的方法解決:
# wget http://ftp.gnu.org/gnu/m4/m4-1.4.9.tar.gz
# tar -zvxf m4-1.4.9.tar.gz
# cd m4-1.4.9/
# ./configure && make && make install
# cd ../
# wget http://ftp.gnu.org/gnu/autoconf/autoconf-2.62.tar.gz
# tar -zvxf autoconf-2.62.tar.gz
# cd autoconf-2.62/
# ./configure && make && make install
./configure --with-php-config=/usr/local/php/bin/php-config make && make install
查看輸出信息會告訴你 redis.so 的那個目錄下,把它複製到PHP的擴展目錄下,在php.ini下開啓擴展目錄
extension_dir = "/usr/local/php/ext" extension=redis.so
個人是 /usr/local/php/ext
加入後重啓WEB服務器,在某個頁面中測試一下是否擴展成功
<?php $redis = new Redis(); $redis->connect('127.0.0.1', 3679) or die('connect failed'); //字符串數據類型設置 $redis->set('mesg', 'hello world!'); var_dump($redis->get('mesg')); //批量設置和獲取 $arrInfo = ['name'=>'sallency', 'age'=>25, sex=>'female']; $redis->mset($arrInfo); $arrInfoKeys = array_keys($arrInfo); var_dump($redis->mget($arrInfoKeys)); ?>
五:PHP負載開發方案
<?php /** * Redis 操做,支持 Master/Slave 的負載集羣 * @author V哥 */ class RedisCluster{ // 是否使用 M/S 的讀寫集羣方案 private $_isUseCluster = false; // Slave 句柄標記 private $_sn = 0; // 服務器鏈接句柄 private $_linkHandle = array( 'master'=>null,// 只支持一臺 Master 'slave'=>array(),// 能夠有多臺 Slave ); /** * 構造函數 * * @param boolean $isUseCluster 是否採用 M/S 方案 */ public function __construct($isUseCluster=false){ $this->_isUseCluster = $isUseCluster; } /** * 鏈接服務器,注意:這裏使用長鏈接,提升效率,但不會自動關閉 * * @param array $config Redis服務器配置 * @param boolean $isMaster 當前添加的服務器是否爲 Master 服務器 * @return boolean */ public function connect($config=array('host'=>'127.0.0.1','port'=>6379), $isMaster=true){ // default port if(!isset($config['port'])){ $config['port'] = 6379; } // 設置 Master 鏈接 if($isMaster){ $this->_linkHandle['master'] = new Redis(); $ret = $this->_linkHandle['master']->pconnect($config['host'],$config['port']); }else{ // 多個 Slave 鏈接 $this->_linkHandle['slave'][$this->_sn] = new Redis(); $ret = $this->_linkHandle['slave'][$this->_sn]->pconnect($config['host'],$config['port']); ++$this->_sn; } return $ret; } /** * 關閉鏈接 * * @param int $flag 關閉選擇 0:關閉 Master 1:關閉 Slave 2:關閉全部 * @return boolean */ public function close($flag=2){ switch($flag){ // 關閉 Master case 0: $this->getRedis()->close(); break; // 關閉 Slave case 1: for($i=0; $i<$this->_sn; ++$i){ $this->_linkHandle['slave'][$i]->close(); } break; // 關閉全部 case 1: $this->getRedis()->close(); for($i=0; $i<$this->_sn; ++$i){ $this->_linkHandle['slave'][$i]->close(); } break; } return true; } /** * 獲得 Redis 原始對象能夠有更多的操做 * * @param boolean $isMaster 返回服務器的類型 true:返回Master false:返回Slave * @param boolean $slaveOne 返回的Slave選擇 true:負載均衡隨機返回一個Slave選擇 false:返回全部的Slave選擇 * @return redis object */ public function getRedis($isMaster=true,$slaveOne=true){ // 只返回 Master if($isMaster){ return $this->_linkHandle['master']; }else{ return $slaveOne ? $this->_getSlaveRedis() : $this->_linkHandle['slave']; } } /** * 寫緩存 * * @param string $key 組存KEY * @param string $value 緩存值 * @param int $expire 過時時間, 0:表示無過時時間 */ public function set($key, $value, $expire=0){ // 永不超時 if($expire == 0){ $ret = $this->getRedis()->set($key, $value); }else{ $ret = $this->getRedis()->setex($key, $expire, $value); } return $ret; } /** * 讀緩存 * * @param string $key 緩存KEY,支持一次取多個 $key = array('key1','key2') * @return string || boolean 失敗返回 false, 成功返回字符串 */ public function get($key){ // 是否一次取多個值 $func = is_array($key) ? 'mGet' : 'get'; // 沒有使用M/S if(! $this->_isUseCluster){ return $this->getRedis()->{$func}($key); } // 使用了 M/S return $this->_getSlaveRedis()->{$func}($key); } /** * 條件形式設置緩存,若是 key 不存時就設置,存在時設置失敗 * * @param string $key 緩存KEY * @param string $value 緩存值 * @return boolean */ public function setnx($key, $value){ return $this->getRedis()->setnx($key, $value); } /** * 刪除緩存 * * @param string || array $key 緩存KEY,支持單個健:"key1" 或多個健:array('key1','key2') * @return int 刪除的健的數量 */ public function remove($key){ // $key => "key1" || array('key1','key2') return $this->getRedis()->delete($key); } /** * 值加加操做,相似 ++$i ,若是 key 不存在時自動設置爲 0 後進行加加操做 * * @param string $key 緩存KEY * @param int $default 操做時的默認值 * @return int 操做後的值 */ public function incr($key,$default=1){ if($default == 1){ return $this->getRedis()->incr($key); }else{ return $this->getRedis()->incrBy($key, $default); } } /** * 值減減操做,相似 --$i ,若是 key 不存在時自動設置爲 0 後進行減減操做 * * @param string $key 緩存KEY * @param int $default 操做時的默認值 * @return int 操做後的值 */ public function decr($key,$default=1){ if($default == 1){ return $this->getRedis()->decr($key); }else{ return $this->getRedis()->decrBy($key, $default); } } /** * 添空當前數據庫 * * @return boolean */ public function clear(){ return $this->getRedis()->flushDB(); } /* =================== 如下私有方法 =================== */ /** * 隨機 HASH 獲得 Redis Slave 服務器句柄 * * @return redis object */ private function _getSlaveRedis(){ // 就一臺 Slave 機直接返回 if($this->_sn <= 1){ return $this->_linkHandle['slave'][0]; } // 隨機 Hash 獲得 Slave 的句柄 $hash = $this->_hashId(mt_rand(), $this->_sn); return $this->_linkHandle['slave'][$hash]; } /** * 根據ID獲得 hash 後 0~m-1 之間的值 * * @param string $id * @param int $m * @return int */ private function _hashId($id,$m=10) { //把字符串K轉換爲 0~m-1 之間的一個值做爲對應記錄的散列地址 $k = md5($id); $l = strlen($k); $b = bin2hex($k); $h = 0; for($i=0;$i<$l;$i++) { //相加模式HASH $h += substr($b,$i*2,2); } $hash = ($h*1)%$m; return $hash; } }// End Class // ================= TEST DEMO ================= // 只有一臺 Redis 的應用 $redis = new RedisCluster(); $redis->connect(array('host'=>'127.0.0.1','port'=>6379)); $redis->set('id',35); var_dump($redis->get('id')); // 有一臺 Master 和 多臺Slave 的集羣應用 $redis = new RedisCluster(true); $redis->connect(array('host'=>'127.0.0.1','port'=>6379), true);// master $redis->connect(array('host'=>'127.0.0.1','port'=>63791), false);// slave 1 $redis->connect(array('host'=>'127.0.0.1','port'=>63792), false);// slave 2 $redis->set('id',100); for($i=1; $i<=100; ++$i){ var_dump($redis->get('id')).PHP_EOL; } // phpRedis 擴展的更多高級操做 $redis = new RedisCluster(); $redis->connect(array('host'=>'127.0.0.1','port'=>6379)); $ret = $redis->getRedis()->ping();// phpRedis 原始API var_dump($ret);
[phpReadis API手冊]
https://github.com/nicolasff/phpredis
1. Redis默認不是以守護進程的方式運行,能夠經過該配置項修改,使用yes啓用守護進程
daemonize no
2. 當Redis以守護進程方式運行時,Redis默認會把pid寫入/var/run/redis.pid文件,能夠經過pidfile指定
pidfile /var/run/redis.pid
3. 指定Redis監聽端口,默認端口爲6379,做者在本身的一篇博文中解釋了爲何選用6379做爲默認端口,由於6379在手機按鍵上MERZ對應的號碼,而MERZ取自意大利歌女Alessia Merz的名字
port 6379
4. 綁定的主機地址
bind 127.0.0.1
5.當 客戶端閒置多長時間後關閉鏈接,若是指定爲0,表示關閉該功能
timeout 300
6. 指定日誌記錄級別,Redis總共支持四個級別:debug、verbose、notice、warning,默認爲verbose
loglevel verbose
7. 日誌記錄方式,默認爲標準輸出,若是配置Redis爲守護進程方式運行,而這裏又配置爲日誌記錄方式爲標準輸出,則日誌將會發送給/dev/null
logfile stdout
8. 設置數據庫的數量,默認數據庫爲0,可使用SELECT <dbid>命令在鏈接上指定數據庫id
databases 16
9. 指定在多長時間內,有多少次更新操做,就將數據同步到數據文件,能夠多個條件配合
save <seconds> <changes>
Redis默認配置文件中提供了三個條件:
save 900 1
save 300 10
save 60 10000
分別表示900秒(15分鐘)內有1個更改,300秒(5分鐘)內有10個更改以及60秒內有10000個更改。
10. 指定存儲至本地數據庫時是否壓縮數據,默認爲yes,Redis採用LZF壓縮,若是爲了節省CPU時間,能夠關閉該選項,但會致使數據庫文件變的巨大
rdbcompression yes
11. 指定本地數據庫文件名,默認值爲dump.rdb
dbfilename dump.rdb
12. 指定本地數據庫存放目錄
dir ./
13. 設置當本機爲slav服務時,設置master服務的IP地址及端口,在Redis啓動時,它會自動從master進行數據同步
slaveof <masterip> <masterport>
14. 當master服務設置了密碼保護時,slav服務鏈接master的密碼
masterauth <master-password>
15. 設置Redis鏈接密碼,若是配置了鏈接密碼,客戶端在鏈接Redis時須要經過AUTH <password>命令提供密碼,默認關閉
requirepass foobared
16. 設置同一時間最大客戶端鏈接數,默認無限制,Redis能夠同時打開的客戶端鏈接數爲Redis進程能夠打開的最大文件描述符數,若是設置 maxclients 0,表示不做限制。當客戶端鏈接數到達限制時,Redis會關閉新的鏈接並向客戶端返回max number of clients reached錯誤信息
maxclients 128
17. 指定Redis最大內存限制,Redis在啓動時會把數據加載到內存中,達到最大內存後,Redis會先嚐試清除已到期或即將到期的Key,當此方法處理 後,仍然到達最大內存設置,將沒法再進行寫入操做,但仍然能夠進行讀取操做。Redis新的vm機制,會把Key存放內存,Value會存放在swap區
maxmemory <bytes>
18. 指定是否在每次更新操做後進行日誌記錄,Redis在默認狀況下是異步的把數據寫入磁盤,若是不開啓,可能會在斷電時致使一段時間內的數據丟失。由於 redis自己同步數據文件是按上面save條件來同步的,因此有的數據會在一段時間內只存在於內存中。默認爲no
appendonly no
19. 指定更新日誌文件名,默認爲appendonly.aof
appendfilename appendonly.aof
20. 指定更新日誌條件,共有3個可選值:
no:表示等操做系統進行數據緩存同步到磁盤(快)
always:表示每次更新操做後手動調用fsync()將數據寫到磁盤(慢,安全)
everysec:表示每秒同步一次(折衷,默認值)
appendfsync everysec
21. 指定是否啓用虛擬內存機制,默認值爲no,簡單的介紹一下,VM機制將數據分頁存放,由Redis將訪問量較少的頁即冷數據swap到磁盤上,訪問多的頁面由磁盤自動換出到內存中(在後面的文章我會仔細分析Redis的VM機制)
vm-enabled no
22. 虛擬內存文件路徑,默認值爲/tmp/redis.swap,不可多個Redis實例共享
vm-swap-file /tmp/redis.swap
23. 將全部大於vm-max-memory的數據存入虛擬內存,不管vm-max-memory設置多小,全部索引數據都是內存存儲的(Redis的索引數據 就是keys),也就是說,當vm-max-memory設置爲0的時候,實際上是全部value都存在於磁盤。默認值爲0
vm-max-memory 0
24. Redis swap文件分紅了不少的page,一個對象能夠保存在多個page上面,但一個page上不能被多個對象共享,vm-page-size是要根據存儲的 數據大小來設定的,做者建議若是存儲不少小對象,page大小最好設置爲32或者64bytes;若是存儲很大大對象,則可使用更大的page,若是不 肯定,就使用默認值
vm-page-size 32
25. 設置swap文件中的page數量,因爲頁表(一種表示頁面空閒或使用的bitmap)是在放在內存中的,,在磁盤上每8個pages將消耗1byte的內存。
vm-pages 134217728
26. 設置訪問swap文件的線程數,最好不要超過機器的核數,若是設置爲0,那麼全部對swap文件的操做都是串行的,可能會形成比較長時間的延遲。默認值爲4
vm-max-threads 4
27. 設置在向客戶端應答時,是否把較小的包合併爲一個包發送,默認爲開啓
glueoutputbuf yes
28. 指定在超過必定的數量或者最大的元素超過某一臨界值時,採用一種特殊的哈希算法
hash-max-zipmap-entries 64
hash-max-zipmap-value 512
29. 指定是否激活重置哈希,默認爲開啓(後面在介紹Redis的哈希算法時具體介紹)
activerehashing yes
30. 指定包含其它的配置文件,能夠在同一主機上多個Redis實例之間使用同一份配置文件,而同時各個實例又擁有本身的特定配置文件
include /path/to/local.conf