亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

重學(xué)Go語(yǔ)言之如何使用Redis

 更新時(shí)間:2023年08月01日 11:00:10   作者:程序員讀書  
Redis是我們開(kāi)發(fā)應(yīng)用程序中很常用的NoSQL數(shù)據(jù)庫(kù),那么在Go語(yǔ)言中要如何連接和操作Redis呢,在這篇文章中,我們就來(lái)一起來(lái)探究一下吧

Redis是我們開(kāi)發(fā)應(yīng)用程序中很常用的NoSQL數(shù)據(jù)庫(kù),作為一款內(nèi)存數(shù)據(jù)庫(kù),Redis也支持?jǐn)?shù)據(jù)持久化(RDBAOF兩種機(jī)制),所以Redis既可以作為數(shù)據(jù)庫(kù)來(lái)使用,也可以作為關(guān)系型數(shù)據(jù)庫(kù)與應(yīng)用程序之間的緩存來(lái)使用。

那么在Go語(yǔ)言中要如何連接和操作Redis呢?在這篇文章中,我們一起來(lái)探究一下!

開(kāi)始

在開(kāi)始操作Redis之前,我們先來(lái)初始化我們的示例項(xiàng)目,接著再安裝連接Redis所需要的第三方庫(kù)。

初始化項(xiàng)目

由于我們接下來(lái)要用到的第三方庫(kù)需要Go Moudles的支持,因此我們先使用go mod init命令初始化我們的示例項(xiàng)目:

go?mod?init?GoTest

Go Redis

由于Go語(yǔ)言的標(biāo)準(zhǔn)庫(kù)并沒(méi)有提供對(duì)Redis的支持,因此就需要使用到Go社區(qū)的第三方庫(kù)。

在這里推薦用Go Reids,該庫(kù)最新版本的Github地址為:

github.com/redis/go-redis

為什么推薦使用Go Redis

推薦使用Go Redis的原因有以下幾點(diǎn):

  • Go RedisRedis官方推薦的第三方庫(kù)。
  • 社區(qū)中使用的人也比較多,而且文檔很齊全。
  • 支持不同種類的Redis客戶端,比如哨兵客戶端,集群客戶端等。
  • 支持所有的Redis版本。

Go Redis安裝

通過(guò)cd命令進(jìn)入項(xiàng)目,執(zhí)行go get命令就可以下載安裝 github.com/redis/go-redis/v9

cd?GoTest
#最新版本為v9
go?get?github.com/redis/go-redis/v9

連接Redis

如同所有的Client/Server應(yīng)用程序一樣,要想操作Redis,就必須先與Redis建立網(wǎng)絡(luò)連接。

簡(jiǎn)單連接

如果是連接部署在內(nèi)部網(wǎng)絡(luò)沒(méi)有密碼的Redis服務(wù)器,只需要IP端口號(hào)(port)就可以連接了:

import?"github.com/redis/go-redis/v9"
opt?:=?&redis.Options{
?Addr:???"localhost:6379",
}
rdb?:=?redis.NewClient(opt)

上面代碼中,可以概述為以下兩個(gè)步驟:

  • 實(shí)例化一個(gè)redis.Options表示連接配置對(duì)象,該對(duì)象當(dāng)前設(shè)置了Addr字段,該 字段表示Redis服務(wù)器的地址。
  • redis.Options對(duì)象傳給redis.NewClient()函數(shù),redis.NewClient()函數(shù)會(huì)返回一個(gè)操作Redis的句柄rdb。

redis.ParseURL

除了自定義redis.Options,另一種方式是調(diào)用redis.ParseURL()函數(shù)以下格式的URL,該函數(shù)會(huì)返回一個(gè)redis.Options對(duì)象:

redis://<user>:<pass>@localhost:6379/<port>

因此我們可以將上面連接Redis的例子改寫為:

import?"github.com/redis/go-redis/v9"
opt,?err?:=?redis.ParseURL("redis://localhost:6379")
if?err?!=?nil?{
?panic(err)
}
rdb?:=?redis.NewClient(opt)

redis.Options

前面我們連接Redis的例子中只用到了redis.OptionsAddr字段:

opt?:=?&redis.Options{
?Addr:???"localhost:6379",
}

除此之外,我們也可以通過(guò)redis.Options的字段配置連接的參數(shù):

opt?:=?&redis.Options{
??//地址與端口
?Addr:???"localhost:6379",
?//數(shù)據(jù)庫(kù),0~16
??DB:0,
??//用戶名,在Redis6.0以上使用
?Username:"test",
?//密碼
?Password:"123456",
?//命令最大重試次數(shù)
??MaxRetries:3,
??//連接超時(shí)
?DialTimeout:3,
??//連接池
?PoolSize:30,
?//最小空閑連接數(shù)
??MinIdleConns:10,
??//最大空閑連接數(shù)
?MaxIdleConns:30,
}

上面列舉的是redis.Options比較常用的字段,實(shí)際上還有很多其他的字段參數(shù),這里就不一一列舉了。

Redis操作

雖然已經(jīng)通過(guò)redis.NewClient()函數(shù)獲取操作Redis的句柄了,但要操作Redis了,還需要?jiǎng)?chuàng)建一個(gè)Context對(duì)象,這是因?yàn)樗械?code>Go Redis操作Redis的方法的第一個(gè)參數(shù)都是Context。

創(chuàng)建Context

Context表示上下文,可以用于超時(shí)控制、數(shù)據(jù)傳遞以及性能監(jiān)控等。

通過(guò)context包的Backgroud()函數(shù)可以創(chuàng)建一個(gè)根Context:

ctx?:=?context.Background()

執(zhí)行已支持的命令

Go Redis為所有的Redis命令都提供了對(duì)應(yīng)的方法,比如SET命令對(duì)應(yīng)的方法為Set,HGET對(duì)應(yīng)的方法為HGet。

因此如果想執(zhí)行對(duì)應(yīng)的Redis命令,直接調(diào)用對(duì)應(yīng)的方法即可,比如:

redis>?set?test?test_value?0
redis>?hset?user:1?id?1?name?小張?gender?男

使用Go Redis執(zhí)行上面命令的代碼為:

rdb.Set(ctx,?"test",?"test_value",?0)
rdb.HSet(ctx,?"user:1",?"id",1,"name",?"小張",?"gender",?"男")

Redis命令執(zhí)行后,一般都有返回值,但是不同的命令的返回值是不一樣的,最簡(jiǎn)單的返回如SET命令返回一個(gè)字符串,HSET命令會(huì)返回一個(gè)整數(shù),而HGETALL會(huì)返回一個(gè)類似結(jié)構(gòu)體的數(shù)據(jù):

package?main
import?(
?"context"
?"fmt"
?"github.com/redis/go-redis/v9"
)
func?main()?{
?rdb?:=?redis.NewClient(&redis.Options{
??Addr:?"localhost:6379",
?})
?ctx?:=?context.Background()
?setCmd?:=?rdb.Set(ctx,?"test",?"test_value",?0)
?hSetCmd?:=?rdb.HSet(ctx,?"user:1",?"id",?1,?"name",?"小張",?"gender",?"男")
?hAllGetCmd?:=?rdb.HGetAll(ctx,?"user:1")
}

在上面的代碼中,我們可以看到在執(zhí)行不同命令時(shí),會(huì)返回不同cmd對(duì)象,比如Set()方法返回一個(gè)StatusCmd對(duì)象,HSet()方法返回IntCmd對(duì)象,而HGetAll()方法返回MapStringStringCmd對(duì)象。

有了cmd對(duì)象后,就可以獲取命令執(zhí)行結(jié)果及錯(cuò)誤信息,有兩種方法,一種是直接調(diào)用Result()方法返回結(jié)果與錯(cuò)誤信息:

fmt.Println(setCmd.Result())
fmt.Println(hSetCmd.Result())
fmt.Println(hAllGetCmd.Result())

另一種方法是分開(kāi)獲得取結(jié)果和錯(cuò)誤信息:

fmt.Println(setCmd.Val(),setCmd.Err())
fmt.Println(hSetCmd.Val(),hSetCmd.Err())
fmt.Println(hAllGetCmd.Val(),hAllGetCmd.Err())

上面兩種輸出方式輸出的的結(jié)果都是:

OK <nil>
3 <nil>
map[gender:男 id:1 name:小張] <nil>

如果我們獲得取一個(gè)不存在的key,此時(shí)命令會(huì)返回一個(gè)特殊的錯(cuò)誤redis.Nil,這是一個(gè)特殊的錯(cuò)誤,用于表示是否獲得取了空值:

val,?err?:=?rdb.Get(ctx,?"key").Result()
switch?{
case?err?==?redis.Nil:
?fmt.Println("key不存在")
case?err?!=?nil:
?fmt.Println("錯(cuò)誤",?err)
case?val?==?"":
?fmt.Println("值是空字符串")
}

執(zhí)行尚未支持的命令

當(dāng)然,有時(shí)候Redis新的版本可能會(huì)添加一些新的命令,而Go Redis還未提供支持,這時(shí)候可以調(diào)用Do方法來(lái)執(zhí)行對(duì)應(yīng)的命令,實(shí)際上Do()方法可以用于執(zhí)行任何命令:

val,?err?:=?rdb.Do(ctx,?"hgetall",?"user:1").Result()
if?err?!=?nil?{
?if?err?==?redis.Nil?{
??fmt.Println("key?does?not?exists")
??return
?}
?panic(err)
}
fmt.Println(val.(MapStringStringCmd))

從上面的例子可以看出Do()執(zhí)行不同的命令后返回值也是不同的。

結(jié)果集映射

有時(shí)候我們查詢Redis后會(huì)返回多個(gè)key-value的結(jié)果集,比如mget、hgethgetall這樣的命令,Go redis提供了Scan()方法可以將查詢回來(lái)的結(jié)果集掃描進(jìn)對(duì)應(yīng)的結(jié)構(gòu)體當(dāng)中:

package?main
import?(
?"context"
?"fmt"
?"github.com/redis/go-redis/v9"
)
type?User?struct?{
?ID?????int????`redis:"id"`
?Name???string?`redis:"name"`
?Gender?string?`redis:"gender"`
}
func?main()?{
?rdb?:=?redis.NewClient(&redis.Options{
??Addr:?"localhost:6379",
?})
?ctx?:=?context.Background()
?var?u?User
?err?:=?rdb.HGetAll(ctx,?"user:1").Scan(&u)
?if?err?!=?nil?{
??panic(err)
?}
?fmt.Println(u)
}

管道

管道(pipeline)允許在一次請(qǐng)求中發(fā)送多條Redis命令,并返回多個(gè)結(jié)果,這樣可以節(jié)省一個(gè)一個(gè)執(zhí)行命令需要付出的往返回時(shí)間:

package?main
import?(
?"context"
?"fmt"
?"github.com/redis/go-redis/v9"
)
func?main()?{
?opt?:=?&redis.Options{
??Addr:?"localhost:6379",
?}
?rdb?:=?redis.NewClient(opt)
?ctx?:=?context.Background()
?cmds,?err?:=?rdb.Pipelined(ctx,?func(pipe?redis.Pipeliner)?error?{
??pipe.Set(ctx,?"test",?"test_value",?0)
??pipe.HSet(ctx,?"user:1",?"id",?1,?"name",?"小張",?"gender",?"男")
??pipe.HGetAll(ctx,?"user:1")
??return?nil
?})
?if?err?!=?nil?{
??panic(err)
?}
?for?_,?cmd?:=?range?cmds?{
??switch?c?:=?cmd.(type)?{
??case?*redis.IntCmd:
???fmt.Println(c.Val())
??case?*redis.StatusCmd:
???fmt.Println(c.Val())
??case?*redis.MapStringStringCmd:
???fmt.Println(c.Val())
??}
?}
}

發(fā)布與訂閱

Go Redis支持發(fā)布與訂閱(Pub/Sub)。

發(fā)布消息

發(fā)布消息的方法為Publish:

package?main
import?(
?"context"
?"fmt"
?"github.com/redis/go-redis/v9"
)
func?main()?{
?rdb?:=?redis.NewClient(&redis.Options{
??Addr:?"localhost:6379",
?})
?ctx?:=?context.Background()
??//向渠道發(fā)送消息
?err?:=?rdb.Publish(ctx,?"mychannel",?"this?is?a?message").Err()
????if?err?!=?nil?{
????????panic(err)
????}
}

訂閱消息

訂閱消息的方法為Subscribe,訂閱之后通過(guò)返回的句柄調(diào)用ReceiveMessage()方法接收消息:

package?main
import?(
?"context"
?"fmt"
?"github.com/redis/go-redis/v9"
)
func?main()?{
?rdb?:=?redis.NewClient(&redis.Options{
??Addr:?"localhost:6379",
?})
?ctx?:=?context.Background()
?pubsub?:=?rdb.Subscribe(ctx,?"mychannel")
?defer?pubsub.Close()
??//接收消息
?for?{
??msg,?err?:=?pubsub.ReceiveMessage(ctx)
??if?err?!=?nil?{
???panic(err)
??}
??fmt.Printf("從%s渠道接受到%s",?msg.Channel,?msg.Payload)
?}
}

小結(jié)

在本文中,我們講解用github.com/redis/go-redis連接和操作Redis的基本操作,其實(shí)這個(gè)庫(kù)支持很多高級(jí)的功能,比如哨兵,集群、分片等功能,以后在別的文章再講解吧。

以上就是重學(xué)Go語(yǔ)言之如何使用Redis的詳細(xì)內(nèi)容,更多關(guān)于Go Redis的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Go 字符串格式化的實(shí)例代碼詳解

    Go 字符串格式化的實(shí)例代碼詳解

    這篇文章主要介紹了Go 字符串格式化的實(shí)例代碼詳解,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-02-02
  • 詳解Go hash算法的支持

    詳解Go hash算法的支持

    這篇文章主要介紹了詳解Go hash算法的支持,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-09-09
  • Go用兩個(gè)協(xié)程交替打印100以內(nèi)的奇偶數(shù)的方法詳解

    Go用兩個(gè)協(xié)程交替打印100以內(nèi)的奇偶數(shù)的方法詳解

    這篇文章主要給大家詳細(xì)介紹了Go用兩個(gè)協(xié)程交替打印100以內(nèi)的奇偶數(shù)的示例代碼,文中給大家介紹了兩個(gè)實(shí)現(xiàn)方法,使用無(wú)緩沖的channel和設(shè)置GOMAXPROCS=1,介紹的非常詳細(xì),需要的朋友可以參考下
    2023-08-08
  • 解決golang中container/list包中的坑

    解決golang中container/list包中的坑

    這篇文章主要介紹了解決golang中container/list包中的坑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2021-04-04
  • Go語(yǔ)言TCP從原理到代碼實(shí)現(xiàn)詳解

    Go語(yǔ)言TCP從原理到代碼實(shí)現(xiàn)詳解

    這篇文章主要為大家介紹了Go語(yǔ)言TCP從原理到代碼實(shí)現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-08-08
  • Go?Redis客戶端使用的兩種對(duì)比

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

    這篇文章主要為大家介紹了Go?Redis客戶端使用對(duì)比詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-07-07
  • Go語(yǔ)言中基本數(shù)據(jù)類型的相互轉(zhuǎn)換詳解

    Go語(yǔ)言中基本數(shù)據(jù)類型的相互轉(zhuǎn)換詳解

    Go在不同類型的變量之間賦值時(shí)需要顯示轉(zhuǎn)換,不能自動(dòng)轉(zhuǎn)換。這篇文章主要和大家介紹了Go語(yǔ)言中基本數(shù)據(jù)類型的相互轉(zhuǎn)換,感興趣的小伙伴可以了解一下
    2022-10-10
  • 如何有效控制Go線程數(shù)實(shí)例探究

    如何有效控制Go線程數(shù)實(shí)例探究

    這篇文章主要為大家介紹了如何有效控制?Go?線程數(shù)的問(wèn)題探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2024-01-01
  • Go語(yǔ)言學(xué)習(xí)之將mp4通過(guò)rtmp推送流媒體服務(wù)的實(shí)現(xiàn)方法

    Go語(yǔ)言學(xué)習(xí)之將mp4通過(guò)rtmp推送流媒體服務(wù)的實(shí)現(xiàn)方法

    對(duì)音視頻一直是小白,決定沉下心來(lái),好好研究一下音視頻知識(shí),下面這篇文章主要給大家介紹了關(guān)于Go語(yǔ)言學(xué)習(xí)之將mp4通過(guò)rtmp推送流媒體服務(wù)的實(shí)現(xiàn)方法,需要的朋友可以參考下
    2022-12-12
  • golang生成RSA公鑰和密鑰的實(shí)現(xiàn)方法

    golang生成RSA公鑰和密鑰的實(shí)現(xiàn)方法

    本文主要介紹了golang生成RSA公鑰和密鑰的實(shí)現(xiàn)方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2024-08-08

最新評(píng)論