redis入门到精通(二) -基本数据类型String_key代表字符偏移量,用什么类型(3)

img
img
img

既有适合小白学习的零基础资料,也有适合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)**

  • 19
    点赞
  • 29
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值