網(wǎng)站首頁 編程語言 正文
介紹
go-redis和redigo底層是通過調(diào)用的萬能 Do 方法實現(xiàn), 但是
redigo:
- 由于輸入是萬能類型所以必須記住每個命令的參數(shù)和返回值情況, 使用起來非常的不友好,
- 參數(shù)類型是萬能類型導(dǎo)致在編譯階段無法檢查參數(shù)類型,
- 每個命令都需要花時間記錄使用方法,參數(shù)個數(shù)等,使用成本高;
go-redis:
- 細(xì)化了每個redis每個命令的功能, 我們只需記住命令,具體的用法直接查看接口的申請就可以了,使用成本低;
- 其次它對數(shù)據(jù)類型按照redis底層的類型進行統(tǒng)一,編譯時就可以幫助檢查參數(shù)類型
- 并且它的響應(yīng)統(tǒng)一采用 Result 的接口返回,確保了返回參數(shù)類型的正確性,對用戶更加友好;
性能對比
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需要顯示申請/關(guān)閉連接,所以總體上二者的性能差異其實不大
Redigo庫
redigo 是Redis數(shù)據(jù)庫的Go客戶端, 操作Redis基本和commands一樣. Redigo命令基本都是通過Do方法來實現(xiàn)的.
Do(ctx context.Context, cmd string, args ...interface{}) (interface{}, error)
雖然調(diào)用Do
函數(shù)萬能參數(shù)可以實現(xiàn)所有的功能,但是使用起來非常的不友好,參數(shù)類型是萬能類型,所以在編譯階段無法檢查參數(shù)類型, 其次每個命令都需要花時間記錄使用方法,參數(shù)個數(shù)等,使用成本高;
演示
演示基本的連接池建立, 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, //最初的連接數(shù)量 MaxActive: 0, //連接池最大連接數(shù)量,(0表示自動定義),按需分配 IdleTimeout: 300, //連接關(guān)閉時間 300秒 (300秒不使用自動關(guān)閉) Dial: func() (redis.Conn, error) { //要連接的redis數(shù)據(jù)庫 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=%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提供了三種對應(yīng)服務(wù)端的客戶端模式,集群,哨兵,和單機模式,三種模式在連接池這一塊都是公用的, 同時還提供了靈活的Hook機制, 其底層實際也是調(diào)用的萬能 Do 方法.
但go-redis細(xì)化了每個redis每個命令的功能, 我們只需記住命令,具體的用法直接查看接口的申請就可以了,使用成本低;其次它對數(shù)據(jù)類型按照redis底層的類型進行統(tǒng)一,編譯時就可以幫助檢查參數(shù)類型, 并且它的響應(yīng)統(tǒng)一采用 Result 的接口返回,確保了返回參數(shù)類型的正確性,對用戶更加友好;
演示
演示基本的連接池建立, 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) { // 新建一個連接池 var pool *redis.Pool pool = &redis.Pool{ MaxIdle: 10, //最初的連接數(shù)量 MaxActive: 1000, //連接池最大連接數(shù)量,(0表示自動定義),按需分配 IdleTimeout: 300, //連接關(guān)閉時間 300秒 (300秒不使用自動關(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() //從連接池,取一個鏈接 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
原文鏈接:https://cloud.tencent.com/developer/article/2056319
相關(guān)推薦
- 2023-06-19 深入了解Golang中Slice切片的使用_Golang
- 2023-07-07 React中useState的setState方法請求了好多次
- 2022-08-16 解決Django?cors跨域問題_python
- 2022-06-02 TensorFlow實現(xiàn)簡單線性回歸_python
- 2023-05-29 tf.nn.conv2d與tf.layers.conv2d的區(qū)別及說明_python
- 2023-05-23 Django?事務(wù)回滾的具體實現(xiàn)_python
- 2022-07-03 關(guān)于python中range()的參數(shù)問題_python
- 2022-06-22 C語言進階教程之函數(shù)指針詳解_C 語言
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支