欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Go?Redis客戶端使用的兩種對(duì)比

 更新時(shí)間:2022年07月27日 11:52:10   作者:Johns  
這篇文章主要為大家介紹了Go?Redis客戶端使用對(duì)比詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪

介紹

go-redis和redigo底層是通過調(diào)用的萬能 Do 方法實(shí)現(xiàn), 但是

redigo:

  • 由于輸入是萬能類型所以必須記住每個(gè)命令的參數(shù)和返回值情況, 使用起來非常的不友好,
  • 參數(shù)類型是萬能類型導(dǎo)致在編譯階段無法檢查參數(shù)類型,
  • 每個(gè)命令都需要花時(shí)間記錄使用方法,參數(shù)個(gè)數(shù)等,使用成本高;

go-redis:

  • 細(xì)化了每個(gè)redis每個(gè)命令的功能, 我們只需記住命令,具體的用法直接查看接口的申請(qǐng)就可以了,使用成本低;
  • 其次它對(duì)數(shù)據(jù)類型按照redis底層的類型進(jìn)行統(tǒng)一,編譯時(shí)就可以幫助檢查參數(shù)類型
  • 并且它的響應(yīng)統(tǒng)一采用 Result 的接口返回,確保了返回參數(shù)類型的正確性,對(duì)用戶更加友好;

性能對(duì)比

BenchmarkRedis/redigo_client_Benchmark-12     31406	     36919 ns/op
BenchmarkRedis/go-redis_client_Benchmark-12   29977	     38152 ns/op
BenchmarkRedis/redigo_client_Benchmark-12     27928	     39923 ns/op
BenchmarkRedis/go-redis_client_Benchmark-12   27127	     46451 ns/op

從上圖可以看出, go-redis雖然每次操作會(huì)比redigo慢10%左右, 但是redigo需要顯示申請(qǐng)/關(guān)閉連接,所以總體上二者的性能差異其實(shí)不大

Redigo庫

redigo 是Redis數(shù)據(jù)庫Go客戶端, 操作Redis基本和commands一樣. Redigo命令基本都是通過Do方法來實(shí)現(xiàn)的.

Do(ctx context.Context, cmd string, args ...interface{}) (interface{}, error)

雖然調(diào)用Do函數(shù)萬能參數(shù)可以實(shí)現(xiàn)所有的功能,但是使用起來非常的不友好,參數(shù)類型是萬能類型,所以在編譯階段無法檢查參數(shù)類型, 其次每個(gè)命令都需要花時(shí)間記錄使用方法,參數(shù)個(gè)數(shù)等,使用成本高;

演示

演示基本的連接池建立, ping, string操作, hash操作, list操作, expire等操作

package main
import (
   "fmt"
   "github.com/gomodule/redigo/redis"
)
func main() {
   // 新建一個(gè)連接池
   var pool *redis.Pool
   pool = &redis.Pool{
      MaxIdle:     10,  //最初的連接數(shù)量
      MaxActive:   0,   //連接池最大連接數(shù)量,(0表示自動(dòng)定義),按需分配
      IdleTimeout: 300, //連接關(guān)閉時(shí)間 300秒 (300秒不使用自動(dòng)關(guān)閉)
      Dial: func() (redis.Conn, error) { //要連接的redis數(shù)據(jù)庫
         return redis.Dial("tcp", "localhost:6379")
      },
   }
   conn := pool.Get() //從連接池,取一個(gè)鏈接
   defer conn.Close()
   // 0. ping正常返回pong, 異常res is nil, err not nil
   res, err := conn.Do("ping")
   fmt.Printf("ping res=%v\n", res)
   if err != nil {
      fmt.Printf("ping err=%v\n", err.Error())
   }
   // string操作
   // set
   res, err = conn.Do("set", "name", "測試001")
   fmt.Printf("set res=%v\n", res)
   if err != nil {
      fmt.Printf("set err=%v\n", err.Error())
   }
   // get
   res, err = redis.String(conn.Do("get", "name"))
   fmt.Printf("get res=%v\n", res)
   if err != nil {
      fmt.Printf("get err=%v\n", err.Error())
   }
   // MSet   MGet
   res, err = conn.Do("MSet", "name", "測試001", "age", 18)
   fmt.Printf("MSet res=%v\n", res)
   if err != nil {
      fmt.Printf("MSet err=%v\n", err.Error())
   }
   r, err := redis.Strings(conn.Do("MGet", "name", "age"))
   fmt.Printf("MGet res=%v\n", r)
   if err != nil {
      fmt.Printf("MGet err=%v\n", err.Error())
   }
   // expire
   res, err = conn.Do("expire", "name", 5)
   fmt.Printf("expire res=%v\n", r)
   if err != nil {
      fmt.Printf("expire err=%v\n", err.Error())
   }
   // list操作
   // lpush lpop
   res, err = conn.Do("lpush", "hobby", "籃球", "足球", "乒乓球")
   fmt.Printf("lpush res=%v\n", r)
   if err != nil {
      fmt.Printf("lpush err=%v\n", err.Error())
   }
   // lpop
   rs, er := conn.Do("lpop", "hobby")
   fmt.Printf("lpop res=%v\n", rs)
   if er != nil {
      fmt.Printf("lpop err=%v\n", er.Error())
   }
   // hash 操作
   // hset
   res, err = conn.Do("HSet", "userinfo", "name", "lqz")
   fmt.Printf("HSet res=%v\n", r)
   if err != nil {
      fmt.Printf("HSet err=%v\n", err.Error())
   }
   // hget
   r4, er4 := conn.Do("HGet", "userinfo", "name")
   fmt.Printf("HGet res=%v\n", r4)
   if er4 != nil {
      fmt.Printf("HGet err=%v\n", er4.Error())
   }
}

go-redis組件介紹和使用

go-redis提供了三種對(duì)應(yīng)服務(wù)端的客戶端模式,集群,哨兵,和單機(jī)模式,三種模式在連接池這一塊都是公用的, 同時(shí)還提供了靈活的Hook機(jī)制, 其底層實(shí)際也是調(diào)用的萬能 Do 方法.

但go-redis細(xì)化了每個(gè)redis每個(gè)命令的功能, 我們只需記住命令,具體的用法直接查看接口的申請(qǐng)就可以了,使用成本低;其次它對(duì)數(shù)據(jù)類型按照redis底層的類型進(jìn)行統(tǒng)一,編譯時(shí)就可以幫助檢查參數(shù)類型, 并且它的響應(yīng)統(tǒng)一采用 Result 的接口返回,確保了返回參數(shù)類型的正確性,對(duì)用戶更加友好;

演示

演示基本的連接池建立, ping, string操作, hash操作, list操作, expire等操作

func main() {
   var rdb = redis2.NewClient(
      &redis2.Options{
         Addr:     "localhost:6379",
         Password: "", DB: 1,
         MinIdleConns: 1,
         PoolSize:     1000,
      })
   ctx := context.Background()
   res, err = rdb.Ping(ctx).Result()
   fmt.Printf("ping res=%v\n", res)
   if err != nil {
      fmt.Printf("ping err=%v\n", err.Error())
   }
   // string操作
   // set
   res, err = rdb.Set(ctx, "name", "測試001", 0).Result()
   fmt.Printf("set res=%v\n", res)
   if err != nil {
      fmt.Printf("set err=%v\n", err.Error())
   }
   // get
   res, err = rdb.Get(ctx, "name").Result()
   fmt.Printf("get res=%v\n", res)
   if err != nil {
      fmt.Printf("get err=%v\n", err.Error())
   }
   // MSet   MGet
   res, err = rdb.MSet(ctx, "name", "測試001", "age", "18").Result()
   fmt.Printf("MSet res=%v\n", res)
   if err != nil {
      fmt.Printf("MSet err=%v\n", err.Error())
   }
   var ret []interface{}
   ret, err = rdb.MGet(ctx, "name", "age").Result()
   fmt.Printf("MGet res=%v\n", ret)
   if err != nil {
      fmt.Printf("MGet err=%v\n", err.Error())
   }
   // expire
   res, err = rdb.Expire(ctx, "name", time.Second).Result()
   fmt.Printf("expire res=%v\n", res)
   if err != nil {
      fmt.Printf("expire err=%v\n", err.Error())
   }
   // list操作
   // lpush lpop
   res, err = rdb.LPush(ctx, "hobby", "籃球", "足球", "乒乓球").Result()
   fmt.Printf("lpush res=%v\n", res)
   if err != nil {
      fmt.Printf("lpush err=%v\n", err.Error())
   }
   // lpop
   rs, err = rdb.LPop(ctx, "hobby").Result()
   fmt.Printf("lpop res=%v\n", rs)
   if er != nil {
      fmt.Printf("lpop err=%v\n", er.Error())
   }
   // hash 操作
   // hset
   res, err = rdb.HSet(ctx, "userinfo", "name", "lqz").Result()
   fmt.Printf("HSet res=%v\n", r)
   if err != nil {
      fmt.Printf("HSet err=%v\n", err.Error())
   }
   // hget
   r4, er4 = rdb.HGet(ctx, "userinfo", "name").Result()
   fmt.Printf("HGet res=%v\n", r4)
   if er4 != nil {
      fmt.Printf("HGet err=%v\n", er4.Error())
   }
}

 性能測試

package main
import (
   "context"
   redis2 "github.com/go-redis/redis/v8"
   "github.com/gomodule/redigo/redis"
   "testing"
   "time"
)
func BenchmarkRedis(b *testing.B) {
   // 新建一個(gè)連接池
   var pool *redis.Pool
   pool = &redis.Pool{
      MaxIdle:     10,   //最初的連接數(shù)量
      MaxActive:   1000, //連接池最大連接數(shù)量,(0表示自動(dòng)定義),按需分配
      IdleTimeout: 300,  //連接關(guān)閉時(shí)間 300秒 (300秒不使用自動(dòng)關(guān)閉)
      Dial: func() (redis.Conn, error) { //要連接的redis數(shù)據(jù)庫
         return redis.Dial("tcp", "localhost:6379")
      },
   }
   var rdb = redis2.NewClient(
      &redis2.Options{
         Addr:         "localhost:6379",
         Password:     "",
         MinIdleConns: 10,
         PoolSize:     1000,
      })
   b.Run("redigo client Benchmark", func(b *testing.B) {
		for j := 0; j < b.N; j++ {
			conn := pool.Get() //從連接池,取一個(gè)鏈接
			conn.Do("set", time.Now().String(), 10000, time.Second)
			conn.Do("get", time.Now().String())
			conn.Close()
		}
	})
	ctx := context.Background()
	b.Run("go-redis client Benchmark", func(b *testing.B) {
		for j := 0; j < b.N; j++ {
			rdb.Set(ctx,  time.Now().String(), 1000, time.Second)
			rdb.Get(ctx,  time.Now().String())
		}
	})
}

結(jié)果輸出

goos: darwin
goarch: amd64
cpu: Intel(R) Core(TM) i7-9750H CPU @ 2.60GHz
BenchmarkRedis
BenchmarkRedis/redigo_client_Benchmark
BenchmarkRedis/redigo_client_Benchmark-12                26386         39110 ns/op
BenchmarkRedis/go-redis_client_Benchmark
BenchmarkRedis/go-redis_client_Benchmark-12              28186         37794 ns/op

以上就是Go Redis客戶端使用對(duì)比的詳細(xì)內(nèi)容,更多關(guān)于Go Redis客戶端對(duì)比的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Go語言中循環(huán)語句使用的示例詳解

    Go語言中循環(huán)語句使用的示例詳解

    在不少實(shí)際問題中有許多具有規(guī)律性的重復(fù)操作,因此在程序中就需要重復(fù)執(zhí)行某些語句。本文將通過示例詳細(xì)為大家講講Go語言中的循環(huán)語句,需要的可以參考一下
    2022-04-04
  • Go語言文件讀寫操作案例詳解

    Go語言文件讀寫操作案例詳解

    這篇文章主要為大家介紹了Go語言文件讀寫操作案例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-09-09
  • 關(guān)于Go語言中的IO操作詳解

    關(guān)于Go語言中的IO操作詳解

    在現(xiàn)代軟件開發(fā)中,高效的輸入輸出(I/O)操作是提高程序性能的關(guān)鍵之一,Go語言提供了豐富的I/O操作接口,使得文件讀寫、網(wǎng)絡(luò)通信等任務(wù)變得簡單而高效,本文介紹了關(guān)于Go語言中的IO操作,需要的朋友可以參考下
    2024-10-10
  • GoFrame基于性能測試得知grpool使用場景

    GoFrame基于性能測試得知grpool使用場景

    這篇文章主要為大家介紹了GoFrame基于性能測試得知grpool使用場景示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-06-06
  • GoZero實(shí)現(xiàn)數(shù)據(jù)庫MySQL單例模式連接的簡單示例

    GoZero實(shí)現(xiàn)數(shù)據(jù)庫MySQL單例模式連接的簡單示例

    在 GoZero 框架中實(shí)現(xiàn)數(shù)據(jù)庫的單例連接可以通過以下步驟來完成,GoZero 使用 gorm 作為默認(rèn)的數(shù)據(jù)庫操作框架,接下來我會(huì)展示一個(gè)簡單的單例模式實(shí)現(xiàn),需要的朋友可以參考下
    2025-02-02
  • gin 獲取post請(qǐng)求的json body操作

    gin 獲取post請(qǐng)求的json body操作

    這篇文章主要介紹了gin 獲取post請(qǐng)求的json body操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧
    2021-03-03
  • golang讀取yaml配置文件的方法實(shí)現(xiàn)

    golang讀取yaml配置文件的方法實(shí)現(xiàn)

    本文主要介紹了golang讀取yaml配置文件的方法實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2024-10-10
  • Go語言單元測試超詳細(xì)解析

    Go語言單元測試超詳細(xì)解析

    本文介紹了了Go語言單元測試超詳細(xì)解析,測試函數(shù)分為函數(shù)的基本測試、函數(shù)的組測試、函數(shù)的子測試,進(jìn)行基準(zhǔn)測試時(shí)往往是對(duì)函數(shù)的算法進(jìn)行測驗(yàn),有時(shí)后一個(gè)算法在測試數(shù)據(jù)的基量不同時(shí)測試出的效果會(huì)不同我們需要對(duì)不同數(shù)量級(jí)的樣本進(jìn)行測試,下文需要的朋友可以參考下
    2022-02-02
  • Go1.21新增cmp包的用法詳解

    Go1.21新增cmp包的用法詳解

    Go?1.21新增的?cmp?包提供了與比較有序值相關(guān)的類型和函數(shù),前幾篇文章講解的?slices?包中的函數(shù)有大量使用到?cmp?包中的函數(shù)和類型,下面我們就來看看cmp包的相關(guān)函數(shù)用法吧
    2023-08-08
  • Golang使用Swagger文檔教程詳解

    Golang使用Swagger文檔教程詳解

    這篇文章主要為大家詳細(xì)介紹Golang使用Swagger文檔教程的相關(guān)知識(shí),文中通過示例代碼講解詳細(xì),具有一定的借鑒價(jià)值,感興趣的小伙伴可以學(xué)習(xí)一下
    2023-12-12

最新評(píng)論