最近由於項目需要,在系統緩存服務部分上了redis,終於有機會在實際開發中玩一下,之前都是自己隨便看看寫寫,很零碎也沒沉澱下來什麼,這次算是一個系統學習和實踐過程的總結。 和Redis有關的基礎知識 Redis是一個開源的分散式NoSql資料庫,可以用來做緩存服務、消息隊列、數據存儲等等,數據類型之 ...
最近由於項目需要,在系統緩存服務部分上了redis,終於有機會在實際開發中玩一下,之前都是自己隨便看看寫寫,很零碎也沒沉澱下來什麼,這次算是一個系統學習和實踐過程的總結。
和Redis有關的基礎知識
Redis是一個開源的分散式NoSql資料庫,可以用來做緩存服務、消息隊列、數據存儲等等,數據類型之豐富,效率之高,簡直逆天!沒有瞭解過的可以移步去問度娘~客戶端之豐富,足可見它的社區有多強大:
其中C#的客戶端就有這麼多:
沒錯,我們的項目里也選擇了最熱門的StackExchange.Redis作為底層服務。
Redis雖然也可以部署在window上,但效率會大打折扣,所以通常都是部署在linux上跑,剛好我在上次部署.net core 項目時創建了一個centos虛擬機,可以直接拿來用,不會配虛擬機的同學可以點這裡。Redis服務部署網上有很多教程,在此就略過了。
在這裡向大家推薦一款redis圖形化操作的客戶端Redis Studio,比Redis Desktop Manager好用太多,誰用誰知道!可以查看運行情況、查看數據和類型、查看剩餘有效時間、刷新數據、刪除數據,甚至可以直接在圖形界面配置redis,再也不用去配置文件里懵逼了。
當然,以上操作都可以在redis服務開啟後用redis-cli工具實現。
基礎操作封裝
緩存的基礎操作無非就是get、set這些,所以統一定義了一個介面:
裡面用StackExchange.Redis提供的API來實現這些操作,代碼太多就不貼出來了。其中對redis的連接做了連接池處理,連接對象ConnectionMultiplexer封裝在一個阻塞隊列裡面,每次讀寫操作的時候去隊列裡面取,用完再放回,在應用啟動的時候會初始化這個連接池。
創建連接的時候有兩種方式,第一種是使用連接字元串的形式,把需要的參數寫在一個字元串中:
第二種是使用ConfigurationOptions對象:
其中EndPoints是redis伺服器的地址,做集群的時候可以寫多個。為了搞清楚裡面參數的含義,從github上clone了一份StackExchange.Redis的源碼來看,非常清晰。在看源碼的過程中發現底層都有記錄redis的詳細運行情況,但都是寫在Stream里,於是自己基於系統的log4net日誌然後根據它的實現重寫了一套TextWriter來實現日誌持久化,用於日後來分析錯誤:
在做泛型操作封裝的時候遇到一個問題:我想把一個複雜對象整存整取。第一個想到的辦法是序列化,但總覺得這樣乾會拖累redis的性能,覺得不爽不想用。然後是用redis的hash類型,但是操作起來非常不方便,而且沒辦法存集合,也pass了,其他的類型翻了源碼看只支持int、string、bool這些,也不行。沒辦法還是序列化吧,然後想起來張善友老師推薦的protobuf,說是性能超級棒,於是就用了,發現它是用Stream來轉化的,莫非這就是它性能高的原因?哪位大神指點下~
用redis自定義session存儲
緩存服務搭起來後就打算把session搬到裡面去,便於做分散式和統一狀態管理。很簡單,重寫一套SessionStateStoreProviderBase就行了,然後跟著園子里焰尾迭的博客《分散式中Redis實現Session終結篇》做了一套,但是發現跑不起來,應該和我的封裝有關,由於時間問題就先放下了,直接上nuget找了一個現成的,看中了Microsoft.Web.RedisSessionStateProvider,想著既然是我軟官方出的應該沒什麼大問題,另外這個SessionStateProvider也是依賴於StackExchange.Redis,與項目中的一致於是果斷下載安裝了。接下來不得不誇一下微軟的細節處理啊,下載完立馬蹦出個readme告訴我安裝結果:
要不然還要研究這個東西怎麼用,按照裡面的提示打開web.config,發現在<system.web>節點下幫我們自動加了一個節點sessionState,裡面定義了和session有關的配置,最重要的是配置模板都給出來了,連數據類型都標記的清清楚楚:
眾所周知微軟一直向他的開發者推崇傻瓜式操作,但這些細節真的是太貼心了,根據自己的redis伺服器信息配置一下關鍵信息就ok了。寫個session測試一下,頁面跑起來了redis裡面也查到session值了(被編碼了),太TM爽啊~100個大寫的贊。。。
最後,把系統中臨時用的HttpContext.Cache和靜態Dictionary全部用redis替換掉,然後build、run、ok。
總結
經過這幾天的實踐算是打開了redis的大門走出了第一步,今後系統開發和運行中肯定還會遇到很多問題,也不是說在程式用redis實現get、set就是學會了,現在接觸到的只是最基礎的東西,後面還要學習一下redis的高級用法,例如pub/sub、master/slave、集群等。
問題
1、序列化那裡心裡還是有梗,有沒有更好的解決方案?而且用protobuf的話要在類名和屬性上打標簽,這個有點憂桑啊~
2、有人說把session放到redis後可以解決session阻塞的問題,測試了一下好像不行啊,哪位大神知道真相的還請指點一二~