Redis 介紹學習

来源:https://www.cnblogs.com/whitebai/archive/2019/10/14/11670631.html
-Advertisement-
Play Games

1、Redis 簡介 Redis 是一個支持數據結構更多的鍵值對資料庫。它的值不僅可以是字元串等基本數據 類型,也可以是類對象,更可以是 Set、List、計數器等高級的數據結構。 Memcached 也可以保存類似於 Set、List 這樣的結構,但是如果說要向 List 中增加元素, Memca ...


1、Redis 簡介

Redis 是一個支持數據結構更多的鍵值對資料庫。它的值不僅可以是字元串等基本數據

類型,也可以是類對象,更可以是 Set、List、計數器等高級的數據結構。
Memcached 也可以保存類似於 Set、List 這樣的結構,但是如果說要向 List 中增加元素, Memcached 則需要把 List 全部元素取出來,然後再把元素增加進去,然後再保存回去,不 僅效率低,而且有併發訪問問題。

Redis 內置的 Set、List 等可以直接支持增加、刪除元素的操作,效率很高,操作是原子的。

Memcached 數據存在記憶體中,memcached 重啟後數據就消失;而 Redis 會把數據持久

化到硬碟中,Redis 重啟後數據還存在。
2、Redis 的安裝
redis for windows >=2.8 的版本支持直接安裝為 windows 服務 https://github.com/MicrosoftArchive/redis
如果下載 msi 自動裝完服務,如果下載 zip 需要按照下麵的方法安裝為服務: https://raw.githubusercontent.com/MSOpenTech/redis/3.0/Windows%20Service%20Documenta tion.md 

3、redis 的優點:

  1. 1)  支持 string、list、set、geo 等複雜的數據結構。

  2. 2)  高命中的數據運行時是在記憶體中,數據最終還是可以保存到磁碟中,這樣伺服器重啟之後數據還在。

  3. 3)  伺服器是單線程的,來自所有客戶端的所有命令都是串列執行的,因此不用擔心併發修改(串列操作當然還是有併發問題)的問題,編程模型簡單;

  4. 4)  支持消息訂閱/通知機制,可以用作消息隊列;

  5. 5)  Key、Value 最大長度允許 512M;

4、redis 的缺點:

  1. 1)  Redis 是單線程的,因此單個 Redis 實例只能使用一個 CPU 核,不能充分發揮伺服器的性能。可以在一臺伺服器上運行多個 Redis 實例,不同實例監聽不同埠,再互相組成集群。

  2. 2)  做緩存性能不如 Memcached;

5、Memcached 的優點:
       1) 多線程,可以充分利用 CPU 多核的性能;

       2) 做緩存性能最高;

6、Memcached 的缺點:

  1. 1)  只能保存鍵值對數據,鍵值對只能是字元串,如果有對象數據只能自己序列化成 json字元串;

  2. 2)  數據保存在記憶體中,重啟後會丟失;

  3. 3)  Key 最大長度 255 個字元,Value 最長 1M。

7、總結
      Memcached 只能當緩存伺服器用,也是最合適的;Redis 不僅可以做緩存伺服器(性能沒有 Memcached 好),還可以存儲業務數據。

8、redis 命令行管理客戶端:

       1)直接啟動 redis 安裝目錄下的 redis-cli 即可。不用管噁心的自動提示。 執行 set name yzk,就是設置鍵值對 name=yzk 執行 get name 就是查找名字是 name 的值;  keys *是查找所有的 key  key *n*是查找所有名字中含有 n 的 key

       2) 和 Redis 一樣,Redis 也是不同系統放到 Redis 中的數據都是不隔離的,因此設定 Key 的 時候也要選擇好 Key。
       3) Redis 伺服器預設建了 16 個資料庫,Redis 的想法是讓大家把不同系統的數據放到不同 的資料庫中。但是建議大家不要這樣用,因為 Redis 是單線程的,不同業務都放到同一個 Redis 實例的話效率就不高,建議放到不同的實例中。

          因此儘量只用預設的 db0資料庫命令行下可以用 select0、select1 這樣的指令切換資料庫,最高為15。試試在不同數據 庫下新建、查詢數據。

       4) 瞭解的常用的幾個命令就可以。所有對數據的操作都可以通過命令行進行,後面講 的.net 操作 Redis 的驅動其實就是對這些命令的封裝。

9、redis GUI 管理客戶端

     GUI 客戶端非常多,個人推薦使用 RedisDesktopManager安裝後點擊【Connect to Redis Server】連接伺服器。展開節點可以看到所有的 Key,雙擊 Key 可以查看 Key 的值。在根節點上點右鍵,選擇 【Console】,這樣就可以輸入命令。

10、.net 操作 Redis

     用 StackExchange.Redis ,而不是 ServiceStack.Redis,因為 StackExchange.Redis 依賴組件 少,而且操作更接近原生的 redis 操作,ServiceStack 封裝的太厲害,而且有過收費的“前科”。

Install-Package StackExchange.Redis
using (ConnectionMultiplexer redis = ConnectionMultiplexer.Connect("localhost:6379")) {

IDatabase db = redis.GetDatabase();//預設是訪問 db0 資料庫,可以通過方法參數指定數 字訪問不同的資料庫

db.StringSet("Name", "abc");

}

支持設置過期時間:db.StringSet("name", "rupeng.com", TimeSpan.FromSeconds(10)) 獲取數據:string s = db.StringGet("Name")如果查不到則返回 null

Redis 里所有方法幾乎都支持非同步,比如 StringGetAsync()、StringSetAsync(),儘量用非同步方法。

      註意看到訪問的參數、返回值是 RedisKey、RedisValue 類型,進行了運算符重載,可以和 string、 byte[]之間進行隱式轉換。

11、Key 操作
       Key 操作:因為 Redis 里所有數據類型都是用 KeyValue 保存,因此 Key 操作針對所有數據類型, KeyDelete(RedisKey key):根據 Key 刪除;KeyExists(RedisKey key)

      判斷 Key 是否存在,儘量不要用, 因為會有併發問題;KeyExpire(RedisKey key, TimeSpan?  expiry)、KeyExpire(RedisKey key, DateTime? expiry)設置過期時間;
12、數據類型

       Redis 支持的數據結構:string、list、set、sortedset、hash、geo(redis 3.2 以上版本)。對應 的 Redis 客戶端里的方法都是 StringXXX、HashXXX、GeoXXX 等方法。

      不同數據類型的操作方 法不能混用,比如不能用 ListXXX 寫入的值用 StringXXX 去讀取或者寫 入等操作。
13、String 類型

      可以用 StringGet、StringSet 來讀寫鍵值對,是基礎操作StringAppend(RedisKey key, RedisValue value):向 Key 的 Value 中附加內容,不存在則新建; 可以用作計數器:db.StringIncrement("count", 2.5);

      給 count 這個計數器增加一個值,如果不存在則從 0 開始加;db.StringDecrement("count",1)計數器減值;獲取還是用 StringGet()獲取字元串類型的 值。比如可以用這個來計算新聞點擊量、點贊量,效率非常高。     

private static string XinWen_Prefix = "WWW_XinWen_";

public async Task<ActionResult> Index(int id)

{
using (ConnectionMultiplexer redis = await ConnectionMultiplexer.ConnectAsync("localhost:6379")) {

IDatabase db = redis.GetDatabase();//預設是訪問 db0 資料庫,可以通過方法參數指定數字訪 問不同的資料庫

//以 ip 地址和文章 id 為 key
string hasClickKey = XinWen_Prefix + Request.UserHostAddress + "_" + id;

//如果之前這個 ip 給這個文章貢獻過點擊量,則不重覆計算點擊量 if(await db.KeyExistsAsync(hasClickKey)==false)
{

await db.StringIncrementAsync(XinWen_Prefix + "XWClickCount" + id, 1); //記錄一下這個 ip 給這個文章貢獻過點擊量,有效期一天 db.StringSet(hasClickKey, "a", TimeSpan.FromDays(1));

}

RedisValue clickCount = await db.StringGetAsync(XinWen_Prefix + "XWClickCount" + id); XinWenModel model = new XinWenModel();
model.ClickCount = Convert.ToInt32(clickCount);
return View(model);

}

return View(); }

14、list 類型

      Redis 中用 List 保存字元串集合。 比如可以把聊天記錄保存到 List 中;商品的物流信息記錄。也  可以當成雙向隊列或者雙向棧用,list 長度是無限。

      ListLeftPush(RedisKey key, RedisValue value)從左側壓棧;RedisValue ListLeftPop(RedisKey key) 從左側彈出;

      ListRightPush(RedisKey key, RedisValue value ) 從右側壓棧;RedisValue ListRightPop(RedisKey key) 從右側彈出;

      RedisValue ListGetByIndex(RedisKey key, long index)獲取 Key 為 key 的 List 中第 index 個元素的值; long ListLength(RedisKey key) 獲取 Key 為 key 的 List 中元素個數;儘量不要用 ListGetByIndex、 ListLength 因為會有併發問題;。

     如果是讀取而不 Pop,則使用 ListRange:RedisValue[] ListRange(RedisKey key, long start = 0, long stop = -1)。不傳 start、end 表示獲取所有數據。指定之後則獲取某個範圍。

      可以把 Redis 的 list 當成消息隊列使用,比如向註冊用戶發送歡迎郵件的工作,可以在註冊的流 程中把要發送郵件的郵箱放到 list 中,另一個程式從 list 中 pop 獲取郵件來發送。

      生產者、消費者模式。把生產過程和消費過程隔離。

15、set 類型
      如大家所知,set 是一種元素不重覆的集合。 SetAdd(RedisKey key, RedisValue value)向 set 中增加元素

      bool SetContains(RedisKey key, RedisValue value) 判斷 set 中是否存在某個元素; long SetLength(RedisKey key) 獲得 set 中元素的個數;

      SetRemove(RedisKey key, RedisValue value)從 set 中刪除元素;

      RedisValue[] SetMembers(RedisKey key)獲取集合中的元素;

      如果使用 set 保存封禁用 id 等,就不用做重覆性判斷了。

      註意 set 不是按照插入順序遍歷的,而是按照自己的一個存儲方式來遍歷,因為沒有保存插入的 順序。

16、sortedset
     如果對於數據遍歷順序有要求,可以使用 sortedset,他會按照打分來進行遍歷。 SortedSetAdd(RedisKey key, RedisValue member, double score) 在 key 這個 sortedset 中增加member,並且給這個 member 打分,如果 member 已經存在,則覆蓋之前的打分;       doubleSortedSetIncrement(RedisKeykey,RedisValuemember,doublevalue) 給key中member這一項增加 value 分;

     double SortedSetDecrement(RedisKey key, RedisValue member, double value):給 key 中 member 這一項減 value 分; 

     SortedSetEntry[] SortedSetRangeByRankWithScores(RedisKey key, long start = 0, long stop = -1,Orderorder=Order.Ascending) 根據排序返回sortedset中的元素以及元素的打分,start、stop用來分頁 查詢、order 用來指定排序規則。

     測試:

db.SortedSetIncrement("Hotwords", "test", 1);

db.SortedSetIncrement("Hotwords", "test", 1); db.SortedSetIncrement("Hotwords", "test", 1); db.SortedSetIncrement("Hotwords", "楊中科", 1); db.SortedSetIncrement("Hotwords", "侯寶林", 1); db.SortedSetIncrement("Hotwords", "侯寶林", 1);
SortedSetEntry[] items = db.SortedSetRangeByRankWithScores("Hotwords"); foreach(var item in items)

{
Console.WriteLine(item.Element+"="+item.Score);

}

 RedisValue[] SortedSetRangeByRank(RedisKey key, long start = 0, long stop = -1, Order order =Order.Ascending) 根據打分排序返回值,可以根據序號查詢其中一部分;

   RedisValue[] SortedSetRangeByScore(RedisKey key, double start = double.NegativeInfinity, doublestop = double.PositiveInfinity, Exclude exclude = Exclude.None, Order order = Order.Ascending, long skip = 0, long take = -1)

    根據打分排序返回值,可以只返回 start- stop 這個範圍的打分;

    sortedset 應用場景:

      1) 用戶每搜一次一個關鍵詞,就給這個關鍵詞加一分;展示熱搜的時候就把前 N 個獲取出來就行了;

      2) 高積分用戶排行榜;

      3) 熱門商品;

       4) 給寶寶投票;

17、Hash

      相當於 value 又是一個“鍵值對集合”或者值是另外一個 Dictionary。 沒想到有什麼應用場景。

18、Geo 類型

Geo 是 Redis 3.2 版本後新增的數據類型,用來保存興趣點(POI,point of interest)的坐標信息。

可以實現計算兩 POI 之間的距離、獲取一個點周邊指定距離的 POI。 下麵添加興趣點數據,”1”、”2”是點的主鍵,點的名稱、地址、電話等存到其他表中。

db.GeoAdd("ShopsGeo", new GeoEntry(116.34039, 39.94218,"1"));

db.GeoAdd("ShopsGeo", new GeoEntry(116.340934, 39.942221, "2"));

db.GeoAdd("ShopsGeo", new GeoEntry(116.341082, 39.941025, "3"));

db.GeoAdd("ShopsGeo", new GeoEntry(116.340848, 39.937758, "4"));

db.GeoAdd("ShopsGeo", new GeoEntry(116.342982, 39.937325, "5"));

db.GeoAdd("ShopsGeo", new GeoEntry(116.340866, 39.936827, "6"));

GeoRemove(RedisKey key, RedisValue member)刪除一個點

查詢兩個 POI 之間的舉例:double? dist = db.GeoDistance("ShopsGeo", "1", "5", GeoUnit.Meters);// 最後一個參數為距離單位根據點的主鍵獲取坐標:GeoPosition? pos = db.GeoPosition("ShopsGeo", "1")

獲取一個 POI 周邊的 POI:

GeoRadiusResult[] results = db.GeoRadius("ShopsGeo", "2", 200, GeoUnit.Meters);//獲取”2”這個周邊 200 米範圍內的 POI
foreach(GeoRadiusResult result in results)
{

Console.WriteLine("Id="+result.Member+",位置"+result.Position+",距離"+result.Distance); }

  獲取一個坐標(這個坐標不一定是 POI)周邊的 POI:

GeoRadiusResult[] results = db.GeoRadius("ShopsGeo", 116.34092, 39.94223, 200, GeoUnit.Meters);// 獲 取(116.34092, 39.94223)這個周邊 200 米範圍內的 POI
foreach(GeoRadiusResult result in results)
{

Console.WriteLine("Id="+result.Member+",位置"+result.Position+",距離"+result.Distance);

}

     Geo Hash 原理:http://www.cnblogs.com/LBSer/p/3310455.html

19、Redis 的批量操作

如果一次性操作很多,會很慢,那麼可以使用批量操作,兩種方式: 1)幾乎所有的操作都支持數組類型,這樣就可以一次性操作多條數據:比如

GeoAdd(RedisKey key, GeoEntry[] values)、SortedSetAdd(RedisKey key, SortedSetEntry[] values) 2) 如果一次性的操作不是簡單的同類型操作,那麼就要使用批量模式:
IBatch batch = db.CreateBatch();
db.GeoAdd("ShopsGeo1", new GeoEntry(116.34039, 39.94218, "1"));

db.StringSet("abc", "123"); batch.Execute();

會把當前連接的 CreateBatch()、Execute()之間的操作一次性提交給伺服器。

20、redis 分散式鎖

     多線程中的 lock 等的作用範圍是當前的程式範圍內的,如果想跨多台伺服器的鎖(盡量避免這樣搞),就要使用分散式鎖。

      

RedisValue token = Environment.MachineName;

//實際項目秒殺此處可換成商品 ID

if (db.LockTake("mylock", token, TimeSpan.FromSeconds(10)))//第三個參數為鎖超時時間,鎖占 用最多 10 秒鐘,超過 10 秒鐘如果還沒有 LockRelease,則也自動釋放鎖,避免了死鎖
{

try {

} finally {

db.LockRelease("mylock", token); }

} else {

Console.WriteLine("獲得鎖失敗"); 
}

  

 
您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 下麵是一些ssh agent的資料簡要摘錄,網路上的相關的文章已經很多了: ssh 推薦的登錄方式是使用私鑰登錄。但是如果生成私鑰的時候,設置了口令(passphrase),每次登錄時需要輸入口令也很麻煩。可以通過 ssh-agent 來管理私鑰,把私鑰載入進記憶體,之後便不用再輸入私鑰。 ssh-a ...
  • 一、基礎知識 1、cortex-m3支持256個中斷,其中包含了16個內核中斷,240個外部中斷 2、STM32只有84個中斷,包括16個內核中斷和68個可屏蔽中斷 3、STM32F103上只有60個可屏蔽中斷(本文重點講解F103) 4、先占優先順序和次占優先順序 先占優先順序(搶占優先順序):搶占優先順序 ...
  • 此題的題意很清晰: 日期相差一天(使用 ); 今天的溫度大於昨天。 ...
  • sql語句如下: SELECT * from A LEFT JOIN B ON A.aID = B.bID ...
  • 一、事務和隔離級別 事務的概念:事務是把對資料庫的一系列操作都看做一個整體,要麼全部成功,要麼全部失敗,利用事務我們可以保證資料庫的完整性,事務具有原子性。 隔離級別:隔離級別定義了事務與事務之間的隔離距離。 臟讀(dirty read):當一個事務讀取另一個事務未提交的修改時,產生臟讀。 不可重覆 ...
  • 一.mysql事件隔離級別 1未提交讀(READUNCOMMITTED) 另一個事務修改了數據,但尚未提交,而本事務中的SELECT會讀到這些未被提交的數據(臟讀)( 隔離級別最低,併發性能高 ) 2..提交讀(READCOMMITTED) 本事務讀取到的是最新的數據(其他事務提交後的)。問題是,在 ...
  • 前言 鏈表提供了高效的節點重排能力, 以及順序性的節點訪問方式, 並且可以通過增刪節點來靈活地調整鏈表的長度。 作為一種常用數據結構, 鏈表內置在很多高級的編程語言裡面, 因為 Redis 使用的 C 語言並沒有內置這種數據結構, 所以 Redis 構建了自己的鏈表實現。 大家可以把Redis的鏈表 ...
  • 1 Kylin是什麼 今天,隨著移動互聯網、物聯網、AI等技術的快速興起,數據成為了所有這些技術背後最重要,也是最有價值的“資產”。如何從數據中獲得有價值的信息?這個問題驅動了相關技術的發展,從最初的基於文件的檢索、分析程式,到數據倉庫理念的誕生,再到基於資料庫的商業智能分析。而現在,這一問題已經變 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...