首頁 > 軟體

Go Redis使用者端使用的兩種對比

2022-07-27 18:01:38

介紹

go-redis和redigo底層是通過呼叫的萬能 Do 方法實現, 但是

redigo:

  • 由於輸入是萬能型別所以必須記住每個命令的引數和返回值情況, 使用起來非常的不友好,
  • 引數型別是萬能型別導致在編譯階段無法檢查引數型別,
  • 每個命令都需要花時間記錄使用方法,引數個數等,使用成本高;

go-redis:

  • 細化了每個redis每個命令的功能, 我們只需記住命令,具體的用法直接檢視介面的申請就可以了,使用成本低;
  • 其次它對資料型別按照redis底層的型別進行統一,編譯時就可以幫助檢查引數型別
  • 並且它的響應統一採用 Result 的介面返回,確保了返回引數型別的正確性,對使用者更加友好;

效能對比

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雖然每次操作會比redigo慢10%左右, 但是redigo需要顯示申請/關閉連線,所以總體上二者的效能差異其實不大

Redigo庫

redigo 是Redis資料庫Go使用者端, 操作Redis基本和commands一樣. Redigo命令基本都是通過Do方法來實現的.

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

雖然呼叫Do函數萬能引數可以實現所有的功能,但是使用起來非常的不友好,引數型別是萬能型別,所以在編譯階段無法檢查引數型別, 其次每個命令都需要花時間記錄使用方法,引數個數等,使用成本高;

演示

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

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

go-redis元件介紹和使用

go-redis提供了三種對應伺服器端的使用者端模式,叢集,哨兵,和單機模式,三種模式在連線池這一塊都是公用的, 同時還提供了靈活的Hook機制, 其底層實際也是呼叫的萬能 Do 方法.

但go-redis細化了每個redis每個命令的功能, 我們只需記住命令,具體的用法直接檢視介面的申請就可以了,使用成本低;其次它對資料型別按照redis底層的型別進行統一,編譯時就可以幫助檢查引數型別, 並且它的響應統一採用 Result 的介面返回,確保了返回引數型別的正確性,對使用者更加友好;

演示

演示基本的連線池建立, 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=%vn", res)
   if err != nil {
      fmt.Printf("ping err=%vn", err.Error())
   }
   // string操作
   // set
   res, err = rdb.Set(ctx, "name", "測試001", 0).Result()
   fmt.Printf("set res=%vn", res)
   if err != nil {
      fmt.Printf("set err=%vn", err.Error())
   }
   // get
   res, err = rdb.Get(ctx, "name").Result()
   fmt.Printf("get res=%vn", res)
   if err != nil {
      fmt.Printf("get err=%vn", err.Error())
   }
   // MSet   MGet
   res, err = rdb.MSet(ctx, "name", "測試001", "age", "18").Result()
   fmt.Printf("MSet res=%vn", res)
   if err != nil {
      fmt.Printf("MSet err=%vn", err.Error())
   }
   var ret []interface{}
   ret, err = rdb.MGet(ctx, "name", "age").Result()
   fmt.Printf("MGet res=%vn", ret)
   if err != nil {
      fmt.Printf("MGet err=%vn", err.Error())
   }
   // expire
   res, err = rdb.Expire(ctx, "name", time.Second).Result()
   fmt.Printf("expire res=%vn", res)
   if err != nil {
      fmt.Printf("expire err=%vn", err.Error())
   }
   // list操作
   // lpush lpop
   res, err = rdb.LPush(ctx, "hobby", "籃球", "足球", "乒乓球").Result()
   fmt.Printf("lpush res=%vn", res)
   if err != nil {
      fmt.Printf("lpush err=%vn", err.Error())
   }
   // lpop
   rs, err = rdb.LPop(ctx, "hobby").Result()
   fmt.Printf("lpop res=%vn", rs)
   if er != nil {
      fmt.Printf("lpop err=%vn", er.Error())
   }
   // hash 操作
   // hset
   res, err = rdb.HSet(ctx, "userinfo", "name", "lqz").Result()
   fmt.Printf("HSet res=%vn", r)
   if err != nil {
      fmt.Printf("HSet err=%vn", err.Error())
   }
   // hget
   r4, er4 = rdb.HGet(ctx, "userinfo", "name").Result()
   fmt.Printf("HGet res=%vn", r4)
   if er4 != nil {
      fmt.Printf("HGet err=%vn", 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) {
   // 新建一個連線池
   var pool *redis.Pool
   pool = &redis.Pool{
      MaxIdle:     10,   //最初的連線數量
      MaxActive:   1000, //連線池最大連線數量,(0表示自動定義),按需分配
      IdleTimeout: 300,  //連線關閉時間 300秒 (300秒不使用自動關閉)
      Dial: func() (redis.Conn, error) { //要連線的redis資料庫
         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() //從連線池,取一個連結
			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())
		}
	})
}

結果輸出

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使用者端使用對比的詳細內容,更多關於Go Redis使用者端對比的資料請關注it145.com其它相關文章!


IT145.com E-mail:sddin#qq.com