PHP中的數據庫三、redis


memcache雖然好用,解決了數據庫遇到高並發時的IO問題,但還有很多問題丞待解決:

1、數據持久性問題,memcache用內存進行存儲,一旦memcache服務器宕機,那么所存儲的數據全部丟失。

2、memcache存儲的數據類型單一,只支持key-value型的數據,要存儲復雜類型的數據,必然需要PHP腳本的大量邏輯操作。


redis基本介紹

redis也是一個內存非關系型數據庫,它擁有memcache在數據存儲上的全部優點,而且在memcache的基礎上(memcache的介紹可以看我的上一篇博文:PHP中的數據庫二、memcache

  • 增加了數據持久性功能,redis用rdb和aof兩種方式實現數據持久性,在服務器突然宕機時也能幾乎保留已存的全部數據。
  • 增加了string(字符串)、set(集合)、sorted_set(有序集合)、hash(哈希)、list(鏈表)數據類型,方便了多類型的存儲和數據庫操作。
  • 增加了安全驗證(可為服務器設置連接密碼)。
  • redis的主從分離等系統更完善(官方開發)。
  • 原生支持發布/訂閱、隊列、緩存等工具。

當然,相比較memcache,它的數據庫操作也較為復雜。


redis的應用場景和安裝

redis除了可以用在memcache能用的地方,它還可以用在:

  • 可以用鏈表來存儲數據,讀取其最新信息。
  • 可以用有序列表存儲數據,讀取其排行榜數據
  • 可以用集合來存儲關注/被關注信息。

在官網(http://redis.io/)下載到它的最新版本,直接解壓,因為redis官方已經編譯過了,直接進行make / make test ,在make install時可以指定其安裝路徑。

安裝完成后,將安裝包里的redis的conf文件mv到安裝目錄的bin目錄下,它是配置和啟動redis所必需的。

除此,安裝目錄文件下bin目錄里還有如下文件。

redis-benchmark //性能測試工具 -n xxx 表示發出xxx條命令用來測試
redis-check-aof //檢查aof日志的工具
redis-check-dump //檢查rbd日志的工具
redis-cli       //客戶端
redis-server    //redis的服務器進程
redis-sentinel //redis哨兵模式的進程

我們用vim打開redis.conf來簡單配置redis服務器。

  • 將daemonize 選項改為yes來后台運行
  • database n 設置一個redis服務器里有n個服務器,默認為0-15共16個
  • port n 來設置redis服務器的監聽端口
  • 設置requirepass yourpassword來設置密碼,客戶端連接后用auth password來通過驗證

我們使用./redis-server ./redis.conf命令來打開redis服務器。

使用./redis-cli [-p port]來連接服務器(默認6379)。


redis的命令

基本(包括string字符串類型)命令

set key value [ex|px n] //設置值[並設置過期時間為n秒/毫秒]
get key //獲取值
del key //刪除值
incby|decby key n //將key值自增或自減n
rename key newkey//覆蓋原來的
select n//選擇第n個數據庫
ttl key //查詢key的過期時間,-1表示永不過期,不存在的為-2
expire key n //設置key的過期時間為n秒  
type key //獲取key的存儲類型
flushdb //清除當前數據庫中的值
shutdown [nosave]//關閉服務器[不存儲]

list(鏈表)命令

lpush/rpush list value1 [value2 value3...] //將value壓入鏈表頭/尾
lpop/rpop list //彈出鏈表頭/尾的值
llen list //獲取鏈表長度

set(集合)命令

sadd set value //往集合中添加value
smembers set //查看集合中的全部數據
srem set value1[value2...]//刪除集合中的元素
sismember set value //判斷value是否是集合中的一個元素

sorted_set(有序集合)命令

zadd sorted_set score1 key1 score2 key2 score3 key3...向有序集合中添加key,並定義其score,集合會用score對其排序
zrange sorted_set a b [withscores]從第a到第b顯示有序列表中的值 b為-1時顯示全部,[顯示各個值的score]
zrank/zrevrank sorted_set key 正序/倒序顯示key在有序集合中的位置
zrem sorted_set key 刪除有序集合中的key
zcard sorted_set [m n]計算有序集合中[score在m到n之間的]一共有多少個

hash(哈希類型)命令

hset hashset key value 設置hash表key的值為value
hget hashset key 獲取hash表的key值
hdel hashset key 刪除hash表中的一個key
hlen hashset 獲取hash表的長度

redis命令繁多,這里只列了一點簡單的,具體命令可以其官網或其中文站http://www.redis.cn/查看翻譯文檔


redis的事務和發布、訂閱

redis中的事務與mysql的類似,只有語句有些不同。

                redis               mysql
開始事務        multi           start transition
                    事務中的query語句
執行事務        exec                commit
回滾事務        discard             roll back

對於並發影響,redis有watch語句控制,被watch語句監測的key值一旦在事務提交前發生變化,則事務自動被取消回滾。

watch key1 [key2...]
unwatch 取消所有監測。

redis原生發布和訂閱功能,它類似於設計模式中的觀察者模式,被訂閱對象一旦發布了新的消息,那么所有訂閱對象都會收到這條消息。使用方式為:

subscribe key //訂閱某個key,如果這個key發布了新的消息,則會收聽到
public key value//發布消息key,值為value,返回值是收到這個消息的人的個數
unsubscribe key //取消監聽
psubscribe key1 key2/pattrn //[根據模式]監聽多個key

redis的數據持久化

redis通過rdb和aof兩種方式實現數據持久化,兩種數據持久化方式都會占用CPU資源,拖慢redis的執行效率,一般兩種模式配合使用。

rdb方式的主要原理就是達到某一寫入條件后把內存中的所有數據的快照保存一份到磁盤上,數據恢復時用數據快照恢復。

aof方式是通過將每條redis執行命令記錄入文本文件,恢復數據時重復執行記錄的命令。

rdb方式實現數據持久化

用save/bgSave命令可以主動使用rdb方式[后台]存儲rdb

修改redis.conf文件進行配置

save m n                    //在m秒內有n次修改即進行一次快照,保存點很重要,一般會配置多個條件,滿足其中之一就保存
stop-writes-on-bgsave-error yes //在進行快照的過程中如果出錯,則停止寫入
rdbcompression yes          //設置進行數據壓縮
rdbchecksum yes             //導入數據時檢查文件是否損壞
dbfilename xxx.rdb          //導出的文件名
dir path                    //導出的文件路徑

aof方式實現數據持久化

aof持久化的問題在於將每條指令都記錄下來,即使是對一個鍵的反復操作,這樣會導致aof文件越來越大,使用aof重寫將會大大減小aof文件的體積,因為它是在最后將數據庫內數據的狀態統一逆化為命令,而不論一個key經過了多少次變化。使用 bgrewrite 命令可手動重寫aof文件。

配置redis.conf文件:

noapppendfsync-on-rewrite yes       //設置導出rdb時停止寫入aof,aof會被寫在內存隊列里,dump rdb 完成后統一進行寫入操作。
appendfsync everysec                //每秒寫入一次
appendfilename                      //path/filename.aof
auto-aof-rewrite-percentage 100     //文件大小增長100%時重寫
auto-aof-rewrite-min-size 64m       //文件至少達到64m時重寫

redis的主從復制

主從復制時,主從都要以自己的.conf文件來啟動服務器。主服務器可以將rdb關閉,以從服務器來產生rdb,加快主服務器的速度。

從服務器復制一個redis6380.conf文件,設置端口,pid存放文件,只讀,主服務器的密碼。

port 6380
pidfile filename
slave-read-only yes
masterauth password

設置完成后,分別用不同的conf文件打開服務器。

考慮到主服務器宕機的情況,我們用sentinel redis哨兵來監測服務器狀態,在主服務器宕機之后做出反應。sentinel是redis集成的,我們只需要將安裝包里的sentinel.conf文件拷貝到redis/bin目錄下,使用redis-sentinel進程文件來啟動服務器即可。

port 26379                                       //sentinel監聽的端口號
daemonize yes                                    //后台啟動進程
sentinel monitor mymaster 192.168.100.211 6379 2 //設置主進程ip和端口號,並設置兩個哨兵發現主服務器長時間無法連接才判定其宕機
sentinel down-after-milliseconds mymaster 30000  //30000毫秒連接不上判定為無法連接
sentinel parallel-syncs mymaster 1               //一個主服務器打開時,同時復制的從服務器數,太大的話會造成服務器瞬間擁堵
sentinel failover-timeout mymaster 900000        //在90000秒內哨兵不再試圖恢復原主服務器

PHP操作redis服務器

安裝好php的redis擴展后(具體看我的博文linux下的PHP),就可以直接使用redis的類函數庫了。

如下是典型的redis應用。

$redis=new Redis();                     //實例化一個Redis對象
$redis->connect('host',port);           //連接redis服務器
$redis->auth('password');               //用密碼認證
$redis->set($key,$value[,$expire_time]);//設置一個值
$content=$redis->get($key);             //獲取值

具體函數使用可以看官方的文檔:https://github.com/phpredis/phpredis

如果您覺得本博文對您有幫助,您可以推薦或關注我,如果您有什么問題,可以在下方留言討論,謝謝。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM