发布于2022年11月4日2年前 Go操作Redis实战 目录安装Redis客户端连接redis基本指令Keys():根据正则获取keysType():获取key对应值得类型Del():删除缓存项Exists():检测缓存项是否存在Expire(),ExpireAt():设置有效期TTL(),PTTL():获取有效期DBSize():查看当前数据库key的数量FlushDB():清空当前数据FlushAll():清空所有数据库字符串(string)类型Set():设置SetEX():设置并指定过期时间SetNX():设置并指定过期时间Get():获取GetRange():字符串截取Incr():增加+1IncrBy():按指定步长增加Decr():减少-1DecrBy():按指定步长减少Append():追加StrLen():获取长度列表(list)类型LPush():将元素压入链表LInsert():在某个位置插入新元素LSet():设置某个元素的值LLen():获取链表元素个数LIndex():获取链表下标对应的元素LRange():获取某个选定范围的元素集从链表左侧弹出数据LRem():根据值移除元素集合(set)类型SAdd():添加元素SPop():随机获取一个元素SRem():删除集合里指定的值SSMembers():获取所有成员SIsMember():判断元素是否在集合中SCard():获取集合元素个数SUnion():并集,SDiff():差集,SInter():交集有序集合(zset)类型ZAdd():添加元素ZIncrBy():增加元素分值ZRange()、ZRevRange():获取根据score排序后的数据段ZRangeByScore()、ZRevRangeByScore():获取score过滤后排序的数据段ZCard():获取元素个数ZCount():获取区间内元素个数ZScore():获取元素的scoreZRank()、ZRevRank():获取某个元素在集合中的排名ZRem():删除元素ZRemRangeByRank():根据排名来删除ZRemRangeByScore():根据分值区间来删除哈希(hash)类型HSet():设置HMset():批量设置HGet():获取某个元素HGetAll():获取全部元素HDel():删除某个元素HExists():判断元素是否存在HLen():获取长度安装Redis客户端Go语言中使用第三方库https://github.com/go-redis/redis连接Redis数据库并进行操作。使用以下命令下载并安装:go get github.com/go-redis/redis/v8 注:导入时指定了版本v8,忽略版本是一个常见错误连接redis说明:Background返回一个非空的Context。 它永远不会被取消,没有值,也没有期限。 它通常在main函数,初始化和测试时使用,并用作传入请求的顶级上下文。示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) //Background返回一个非空的Context。 它永远不会被取消,没有值,也没有期限。 //它通常在main函数,初始化和测试时使用,并用作传入请求的顶级上下文。 var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) pong, err := rdb.Ping(ctx).Result() if err != nil { fmt.Printf("连接redis出错,错误信息:%v", err) } fmt.Println("成功连接redis") } 基本指令Keys():根据正则获取keys示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) //*表示获取所有的key keys, err := rdb.Keys(ctx, "*").Result() if err != nil { panic(err) } fmt.Println(keys) } Type():获取key对应值得类型Type()方法用户获取一个key对应值的类型示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) vType, err := rdb.Type(ctx, "key").Result() if err != nil { panic(err) } fmt.Println(vType) //string } Del():删除缓存项示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) n, err := rdb.Del(ctx, "key1", "key2").Result() if err != nil { panic(err) } fmt.Printf("成功删除了 %v 个\n", n) } Exists():检测缓存项是否存在Exists()方法用于检测某个key是否存在package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) n, err := rdb.Exists(ctx, "key1").Result() if err != nil { panic(err) } if n > 0 { fmt.Println("存在") } else { fmt.Println("不存在") } } 注:Exists()方法可以传入多个key,返回的第一个结果表示存在的key的数量,不过工作中我们一般不同时判断多个key是否存在,一般就判断一个key,所以判断是否大于0即可,如果判断判断传入的多个key是否都存在,则返回的结果的值要和传入的key的数量相等Expire(),ExpireAt():设置有效期需要在设置好了缓存项后,在设置有效期Expire()方法是设置某个时间段(time.Duration)后过期,ExpireAt()方法是在某个时间点(time.Time)过期失效package main import ( "context" "fmt" "github.com/go-redis/redis/v8" "time" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) res, err := rdb.Expire(ctx, "key", time.Minute * 2).Result() if err != nil { panic(err) } if res { fmt.Println("设置成功") } else { fmt.Println("设置失败") } res, err = rdb.ExpireAt(ctx, "key2", time.Now()).Result() if err != nil { panic(err) } if res { fmt.Println("设置成功") } else { fmt.Println("设置失败") } } TTL(),PTTL():获取有效期TTL()方法可以获取某个键的剩余有效期示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" "time" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) //设置一分钟的有效期 rdb.Expire(ctx, "key", time.Minute) //获取剩余有效期,单位:秒(s) ttl, err := rdb.TTL(ctx, "key").Result() if err != nil { panic(err) } fmt.Println(ttl) //获取剩余有效期,单位:毫秒(ms) pttl, err := rdb.PTTL(ctx, "key").Result() if err != nil { panic(err) } fmt.Println(pttl) } DBSize():查看当前数据库key的数量示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) num, err := rdb.DBSize(ctx).Result() if err != nil { panic(err) } fmt.Printf("数据库有 %v 个缓存项\n", num) } FlushDB():清空当前数据示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) //清空当前数据库,因为连接的是索引为0的数据库,所以清空的就是0号数据库 res, err := rdb.FlushDB(ctx).Result() if err != nil { panic(err) } fmt.Println(res)//OK } FlushAll():清空所有数据库示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) res, err := rdb.FlushAll(ctx).Result() if err != nil { panic(err) } fmt.Println(res)//OK } 字符串(string)类型Set():设置仅仅支持字符串(包含数字)操作,不支持内置数据编码功能。如果需要存储Go的非字符串类型,需要提前手动序列化,获取时再反序列化。示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" "time" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) _, err := rdb.Ping(ctx).Result() //fmt.Println(pong, err) if err != nil { fmt.Printf("连接redis出错,错误信息:%v", err) return } //Set方法的最后一个参数表示过期时间,0表示永不过期 err = rdb.Set(ctx, "key1", "value1", 0).Err() if err != nil { panic(err) } //key2将会在两分钟后过期失效 err = rdb.Set(ctx, "key2", "value2", time.Minute * 2).Err() if err != nil { panic(err) } } SetEX():设置并指定过期时间设置键的同时,设置过期时间示例:package main import ( "context" "github.com/go-redis/redis/v8" "time" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) err := rdb.SetEX(ctx, "key", "value", time.Hour * 2).Err() if err != nil { panic(err) } } SetNX():设置并指定过期时间注:SetNX()与SetEX()的区别是,SexNX()仅当key不存在的时候才设置,如果key已经存在则不做任何操作,而SetEX()方法不管该key是否已经存在缓存中直接覆盖介绍了SetNX()与SetEX()的区别后,还有一点需要注意的时候,如果我们想知道我们调用SetNX()是否设置成功了,可以接着调用Result()方法,返回的第一个值表示是否设置成功了,如果返回false,说明缓存Key已经存在,此次操作虽然没有错误,但是是没有起任何效果的。如果返回true,表示在此之前key是不存在缓存中的,操作是成功的示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" "time" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) res, err := rdb.SetNX(ctx, "key", "value", time.Minute).Result() if err != nil { panic(err) } if res { fmt.Println("设置成功") } else { fmt.Printf("key已经存在缓存中,设置失败") } } Get():获取如果要获取的key在缓存中并不存在,Get()方法将会返回redis.Nil示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) err := rdb.Set(ctx, "key", "value", 0).Err() if err != nil { panic(err) } val, err := rdb.Get(ctx, "key").Result() if err != nil { panic(err) } fmt.Printf("key: %v\n", val) val2, err := rdb.Get(ctx, "key-not-exist").Result() if err == redis.Nil { fmt.Println("key不存在") } else if err != nil { panic(err) } else { fmt.Printf("值为: %v\n", val2) } } GetRange():字符串截取GetRange()方法可以用来截取字符串的部分内容,第二个参数是下标索引的开始位置,第三个参数是下标索引的结束位置(不是要截取的长度)示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) err := rdb.Set(ctx, "key", "Hello World!", 0).Err() if err != nil { panic(err) } val, err := rdb.GetRange(ctx, "key", 1, 4).Result() if err != nil { panic(err) } fmt.Printf("key: %v\n", val) //截取到的内容为: ello } 注:即使key不存在,调用GetRange()也不会报错,只是返回的截取结果是空"",可以使用fmt.Printf("%q\n", val)来打印测试Incr():增加+1Incr()、IncrBy()都是操作数字,对数字进行增加的操作,incr是执行原子加1操作,incrBy是增加指定的数所谓原子操作是指不会被线程调度机制打断的操作:这种操作一旦开始,就一直运行到结束,中间不会有任何context witch(切换到另一个线程).(1)在单线程中,能够在单条指令中完成的操作都可以认为是“原子操作”,因为中断只能发生于指令之间。(2)在多线程中,不能被其它进程(线程)打算的操作叫原子操作。Redis单命令的原子性主要得益于Redis的单线程示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) val, err := rdb.Incr(ctx, "number").Result() if err != nil { panic(err) } fmt.Printf("key当前的值为: %v\n", val) } IncrBy():按指定步长增加示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) val, err := rdb.IncrBy(ctx, "number", 12).Result() if err != nil { panic(err) } fmt.Printf("key当前的值为: %v\n", val) } Decr():减少-1Decr()和DecrBy()方法是对数字进行减的操作,和Incr正好相反示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) val, err := rdb.Decr(ctx, "number").Result() if err != nil { panic(err) } fmt.Printf("key当前的值为: %v\n", val) } DecrBy():按指定步长减少示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) val, err := rdb.DecrBy(ctx, "number", 12).Result() if err != nil { panic(err) } fmt.Printf("key当前的值为: %v\n", val) } Append():追加Append()表示往字符串后面追加元素,返回值是字符串的总长度package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background() func main() { rdb := redis.NewClient(&redis.Options{ Addr: "172.16.147.128:6379", Password: "", DB: 0, }) err := rdb.Set(ctx, "key", "hello", 0).Err() if err != nil { panic(err) } length, err := rdb.Append(ctx, "key", " world!").Result() if err != nil { panic(err) } fmt.Printf("当前缓存key的长度为: %v\n", length) //12 val, err := rdb.Get(ctx, "key").Result() if err != nil { panic(err) } fmt.Printf("当前缓存key的值为: %v\n", val) //hello world! } StrLen():获取长度StrLen()方法可以获取字符串的长度示例:package main import ( "context" "fmt" "github.com/go-redis/redis/v8" ) var ctx = context.Background()
创建帐户或登录后发表意见