既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Go语言开发知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新
MSET key value [key value ...]
同时设置多个key-value对。如果某个key已经存在,那么mset会使用新值覆盖旧值。
MSET是原子操作,所有给定的key都会在同一时间内被设置。
1.4.2 返回值
总是返回OK,MSET不会失败
1.4.3 复杂度
O(N),N为用户给定的字符串键数量。
1.4.4 使用实例
> MSET id "10086" name "zhangsan"
OK
> MGET id name
1) 10086
2) zhangsan
// MSET 覆盖旧值例子
> SET id "10086"
OK
> MSET id "1008611"
OK
> GET id
1008611
1.5 MSETNX:只在键不存在的情况下一次为多个字符串设置值
1.5.1 命令格式
MSETNX key value [key value ...]
同时设置多个key-value对。如果某个key已经存在,那么msetnx会放弃所有键值对设置操作。
1.5.2 返回值
成功时返回1,失败时返回0
1.5.3 复杂度
O(N),N为用户给定的字符串键数量。
1.5.4 使用实例
> msetnx id 10086 name zhangsan
1
> MGET id name
10086
zhangsan
> msetnx id 1008611 name lisi sex nan
0
> mget id name sex
10086
zhangsan
null
1.6 MGET:一次获取多个字符串键的值
1.6.1 命令格式
获取多个字符串键对应的值。
1.6.2 返回值
所有给定key的值。
1.6.3 复杂度
O(N) , N 为给定 key 的数量。
1.7 STRLEN: 获取字符串的字节长度
1.7.1 命令格式
STRLEN key
返回key所存储得字符串长度,当key存储不是字符串时,返回错误。
1.7.2 返回值
返回key所存储得字符串长度,当key存储不是字符串时,返回错误。
1.7.3复杂度
O(1)
1.7.4 使用实例
> set key "hello world"
OK
> strlen key
11
//不存在的key
> strlen noexist
0
1.8 GETRANGE:获取字符串值的指定范围上的内容
字符串是由连续的字节组成的,所以每个字节都有相对应的索引。redis为字符串提供了索引操作命令,允许用户通过正向/负向索引进行处理
- 正向索引以0为第一个字节,从开头向结尾不断递增。
- 负向索引以-1为最后一个字节,向开头递减。
例如字符串hello world的索引如图
1.8.1 命令格式
GETRANGE key start end
通过GETRANGE命令用户可以获取字符串值从start索引,直到end索引为止。此命令的偏移是闭区间索引范围,也就是说start索引和end索引也包含在命令返回内容中。
1.8.2 返回值
截取的子字符串。
1.8.3复杂度
O(N),N为要返回的字符串长度。
1.8.4 使用实例
> set key "hello redis!"
OK
// 截取0-4的字符串
> getrange key 0 4
hello
// 获取失败
> getrange key -1 -5
// 负索引
> getrange key -3 -1
is!
// 从第一个到最后一个
> getrange key 0 -1
hello redis!
// 超出返回忽略
> getrange key 0 100
hello redis!
1.10 SETRANGE:对字符串值指定范围进行设置
1.10.1 命令格式
SETRANGE key offset value
从偏移量offset开始使用value覆盖原有key。
- 如果key不存在当作空字符串处理。
- 如果key原来的字符串比偏移量小,那么原字符和偏移量之间的空白将用零字节(’\x00’)填充。
- 最大偏移量是2^29-1(536870911),因为redis字符串的大小限制在512M以内。
- 当生成一个很长字符串时,redis需要分配内存空间,所以有时会造成服务器阻塞。
1.10.2 返回值
被 SETRANGE 修改之后,字符串的长度。
1.10.3复杂度
对于小的字符串,平均复杂度为O(1),否则为O(M),M为value的长度。
1.10.4 使用实例
> set key "hello world"
OK
> setrange key 6 redis
11
> get key
hello redis
// 对不存在的key进行setrange
> setrange noexist 5 'redis'
10
> get noexist
redis
1.11 APPEND:追加新的内容到值得末尾
1.11.1 命令格式
APPEND key value
如果key已经存在并且是一个字符串,将value追加到key原来的值的末尾。
1.11.2 返回值
返回追加value后key的字符串的长度。
1.11.3复杂度
平均O(1)
1.11.4 使用实例
···
exists key
0
// 对不存在的key进行append,等同于set
append key “hello”
5
// 对与存在key进行append
append key " world"
11
get key
hello world
···
1.12 INCRBY、DECRBY、INCR、DECR:对整数进行加减法操作
在redis中如果用户将一个值存储到字符串键中,redis会对这个值进行检测,如果这个值能够被解释为以下两种其中一个,那么redis就会把这个值当作数字来处理:
- 整数类型:能够使用C的long long int类型存储的整数,大多数系统中,这种类型存储的都是64位长度的有符号整数。
- 浮点数类型:能够使用C的long double类型存储的浮点数,在大多数系统中,这种类型存储的都是128位长度的有符号浮点数。
当字符串键可以被解释为整数时,就可以使用INCRBY和DECRBY命令对整数进行加、减法操作。
1.12.1 命令格式
INCRBY key increment
DECRBY key decrement
INCR key
DECR key
如果key所对应的类型不是整形,那么会返回一个错误。
- INCRBY:对key所存储的值加上增量,如果key不存在,那么key会被初始化为0再执行INCRBY。
- DECRBY:对key所存储的值减去减量,如果key不存在,那么key会被初始化为0再执行DECRBY。
- INCR:对key所存储的值加上1,如果key不存在,那么key会被初始化为0再执行INCR。
- DECR:对key所存储的值减去1,如果key不存在,那么key会被初始化为0再执行DECR。
时间复杂度为O(1)
1.13 INCRBYFLOAT:对浮点数进行加法操作
1.13.1 命令格式
INCRBYFLOAT key increment
INCRBYFLOAT既可以用于浮点数,也可以用于整数。
在使用INCRBYFLOAT处理浮点数最多只会保留计算小数点后17位,超出的将被截断。
时间复杂度位O(1)
1.14
二、 应用
2.1 缓存
redis是内存数据库,所以速度会快得多,而且redis的字符串键不仅可以存储文本数据还可以存储二进制数据,所以可以用来缓存热门图片,从而提高热门图片的速度。
package main
import (
"context"
"fmt"
"github.com/go-redis/redis/v8"
"io"
"os"
)
func main() {
rdb := redis.NewClient(&redis.Options{
Addr: "127.0.0.1:6379",
DB: 0,
})
ctx := context.Background()
rdb.FlushDB(ctx)
file, err := os.Open("D://workspace//myOpenSource//go-example//go-redis//01-string//cache//redis-logo.jpg")
defer file.Close()
if err != nil {
panic(err)
}
data, err := io.ReadAll(file)
if err != nil {
panic(err)
}
rdb.Set(ctx, "redis-logo.jpg", data, -1)
fmt.Println(rdb.Get(ctx, "redis-logo.jpg"))
}
2.2 锁
锁是一种同步机制,用于保证一项资源在任何时候只能被一个进程使用,如果有其他进程想要使用相同的资源时必须等待,直到正在使用的资源进程放弃使用所有全。
锁通常会实现获取(acquire)和释放(release)两种操作:
- 获取(acquire):用于取得资源的独占使用权。在任何时候最多只能有一个进程取得锁,我们把成功取得锁的这个进程称为锁的持有者。在锁已经被持有的情况下,所有尝试获取锁的操作都会失败。
- 释放(release): 用于放弃资源的独占使用权,一般由锁的持有者调用。在锁被释放后其他进程可以再尝试获取锁。
下列代码展示GO实现锁
package main
import (
"context"
"fmt"
"github.com/go-redis/redis/v8"
)
type Lock struct {
ctx context.Context
rdb *redis.Client
key string
}
func New(key string) *Lock {
lock := &Lock{
ctx: context.Background(),
rdb: redis.NewClient(&redis.Options{
Addr: "127.0.0.1:6379",
DB: 0,
}),
key: key,
}
return lock
}
func (self *Lock) acquire() bool {
ok, _ := self.rdb.SetNX(self.ctx, self.key, "lock", -1).Result()
return ok
}
func (self *Lock) release() bool {
![img](https://img-blog.csdnimg.cn/img_convert/ae8601d6c94fe9f1f47ede6771ba2d41.png)
![img](https://img-blog.csdnimg.cn/img_convert/9481042c2a7a7065fa753b3dc052fec1.png)
![img](https://img-blog.csdnimg.cn/img_convert/33312eee8d8d38d8dfef42483c74476a.png)
**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Go语言开发知识点,真正体系化!**
**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**
**[如果你需要这些资料,可以戳这里获取](https://bbs.csdn.net/topics/618658159)**
f *Lock) acquire() bool {
ok, _ := self.rdb.SetNX(self.ctx, self.key, "lock", -1).Result()
return ok
}
func (self *Lock) release() bool {
[外链图片转存中...(img-Lf7bXeGl-1715909372840)]
[外链图片转存中...(img-XjE5tfGX-1715909372840)]
[外链图片转存中...(img-0Mr88RG9-1715909372841)]
**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Go语言开发知识点,真正体系化!**
**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**
**[如果你需要这些资料,可以戳这里获取](https://bbs.csdn.net/topics/618658159)**