使用Redis開發應用程序是一個很愉快的過程,可是就像其餘技術同樣,基於Redis的應用程序設計你一樣須要牢記幾點。在以前,你可能已經對關係型數據庫開發的那一整個套路瞭然如胸,而基於Redis的應用程序開發也有許多類似的地方,可是你必須牢記如下兩點——Redis是個內存數據庫,同時它是單線程的。所以,在使用Redis時,你須要注意如下幾點:
1. 掌控儲存在Redis中的全部鍵
數據庫的主要功能是儲存數據,可是對於開發者來講,由於應用程序需求或者數據使用方法的改變,忽略存儲在數據庫中的某些數據是很是正常的,在Redis中一樣如此。你可能忽視期滿某些鍵,也可能由於應用程序的某個模塊棄用而忘掉這些數據。
不管哪一種狀況,Redis都存儲了一些再也不使用的數據,無緣無故的佔用了一些空間。Redis的弱結構數據模式讓集中儲存的內容很難被弄清,除非你爲鍵使用一套很是成熟的命名法則。使用合適的命名方法會簡化你的數據庫管理,當你經過你的應用程序或者服務作鍵的命名空間時(一般狀況下是使用冒號來劃分鍵名),你就能夠在數據遷移、轉換或者刪除時輕鬆的識別。
Redis另外一個常見用例是做爲熱數據項做的第二數據存儲,大部分的數據被保存在其餘的數據庫中,好比PostgreSQL或MongoDB。在這些用例中,當數據從主存儲移除時,開發者常常會忘記刪除Redis中對應的數據。這種存在跨數據存儲的狀況下,一般須要作級聯刪除,這種狀況下,能夠經過在Redis配置保存特定數據項的全部識別符來實現,從而保證數據在主數據庫被刪除後,系統會調用一個清理程序來刪除全部相關副本和信息。
2. 控制全部鍵名的長度
在上文咱們說過要使用合適的命名規則,而且添加前綴來識別數據走向,所以這一條看起來彷佛與之違背。可是,請別忘記,Redis是個內存數據庫,鍵越短你須要的空間就越少。理所固然,當數據庫中擁有數百萬或者數十億鍵時,鍵名的長度將影響重大。
舉個例子:在一個32位的Redis服務器上,若是儲存一百萬個鍵,每一個值的長度是32-character,那麼在使用6-character長度鍵名時,將會消耗大約96MB的空間,可是若是使用12-character長度的鍵名時,空間消耗則會提高至111MB左右。隨着鍵的增多,15%的額外開銷將產生重大的影響。
3. 使用合適的數據結構
不論是內存使用或者是性能,有的時候數據結構將產生很大的影響,下面是一些能夠參考的最佳實踐:
取代將數據存儲爲數千(或者數百萬)獨立的字符串,能夠考慮使用哈希數據結構將相關數據進行分組。哈希表是很是有效率的,而且能夠減小你的內存使用;同時,哈希還更有益於細節抽象和代碼可讀。
合適時候,使用list代替set。若是你不須要使用set特性,List在使用更少內存的狀況下能夠提供比set更快的速度。
Sorted sets是最昂貴的數據結構,不論是內存消耗仍是基本操做的複雜性。若是你只是須要一個查詢記錄的途徑,並不在乎排序這樣的屬性,那麼輕建議使用哈希表。
Redis中一個常常被忽視的功能就是bitmaps或者bitsets(V2.2以後)。Bitsets容許你在Redis值上執行多個bit-level操做,好比一些輕量級的分析。
4. 使用SCAN時別使用鍵
從Redis v2.8開始,SCAN命令已經可用,它容許使用遊標從keyspace中檢索鍵。對比KEYS命令,雖然SCAN沒法一次性返回全部匹配結果,可是卻規避了阻塞系統這個高風險,從而也讓一些操做能夠放在主節點上執行。
須要注意的是,SCAN 命令是一個基於遊標的迭代器。SCAN 命令每次被調用以後, 都會向用戶返回一個新的遊標,用戶在下次迭代時須要使用這個新遊標做爲 SCAN 命令的遊標參數, 以此來延續以前的迭代過程。同時,使用SCAN,用戶還可使用keyname模式和count選項對命令進行調整。
SCAN相關命令還包括SSCAN 命令、HSCAN 命令和 ZSCAN 命令,分別用於集合、哈希鍵及有續集等。
5. 使用服務器端Lua腳本
在Redis使用過程當中,Lua腳本的支持無疑給開發者提供一個很是友好的開發環境,從而大幅度解放用戶的創造力。若是使用得當,Lua腳本能夠給性能和資源消耗帶來很是大的改善。取代將數據傳送給CPU,腳本容許你在最接近數據的地方執行邏輯,從而減小網絡延時和數據的冗餘傳輸。
在Redis中,Lua一個很是經典的用例就是數據過濾或者將數據聚合到應用程序。經過將處理工做流封裝到一個腳本中,你只須要調用它就能夠在更短的時間內使用不多的資源來獲取一個更小的答案。
專家提示:Lua確實很是棒,可是一樣也存在一些問題,好比很難進行錯誤報告和處理。一個明智的方法就是使用Redis的Pub/Sub功能,而且讓腳本經過專用信道來推送日誌消息。而後創建一個訂閱者進程,並進行相應的處理。數據庫