# 扩展库Redis > 2021年11月18日,腾讯云和阿里云均支持 > 使用腾讯云node12和redis,务必仔细阅读此文档:[keepRunningAfterReturn](uniCloud/cf-functions.md?id=keep-running) Redis是一个基于key/value的内存数据库。在项目中通常作为MongoDB等磁盘数据库的补充来搭配使用。 相对于磁盘数据库,Redis的核心优势是快。因为操作内存要比磁盘快的多,并且Redis只支持key/value数据,读写都很快。但Redis没有磁盘数据库丰富的查询等功能。 Redis一般需要与MongoDB搭配使用,MongoDB做主存储,Redis缓存部分数据应对高性能需求场景。 在uniCloud中,Redis还有一个特点,是它按容量和使用时长计费,访问它不耗费云数据库的读写次数。 Redis常见使用场景: - 缓存高频数据,比如首页列表、banner列表、热门排行。MongoDB数据更新后同步给Redis,这些频繁的请求就不再查询MongoDB数据库 - 秒杀、抢购。短时间大量并发可能发生超卖,此时必须使用Redis解决 - ip黑名单屏蔽。希望较快封杀某些ip请求,缓解MongoDB数据库压力。 - 其他数据库操作速度不满足需求的场景 ## 开通Redis扩展库@buy 参考[开通redis](uniCloud/redis-buy.md) ## 为云函数启用redis扩展库@use-in-function Redis的sdk体积不小,没有内置在云函数中。在需要Redis的云函数里,开发者需手动配置Redis扩展库。(Redis没有免费试用,需购买才可以使用) - HBuilderX 3.4起提供了可视化界面,新建云函数/云对象时可选择Redis扩展库,或者在已有的云函数目录点右键选择“管理公共模块或扩展库依赖” - HBuilderX 3.4以前,没有可视化界面,需要开发者手动在云函数/云对象的package.json内添加云函数的扩展库(如果云函数目录下没有package.json,可以通过在云函数目录下执行`npm init -y`来生成) 下面是一个开启了redis扩展库的云函数的package.json示例,**注意不可有注释,以下文件内容中的注释仅为说明,如果拷贝此文件,切记去除注释** ```js { "name": "redis-test", "version": "1.0.0", "description": "", "main": "index.js", "extensions": { "uni-cloud-redis": {} // 配置为此云函数开启redis扩展库,值为空对象留作后续追加参数,暂无内容 }, "author": "" } ``` ## 云函数中调用Redis ```js // 云函数中调用Redis示例 'use strict'; const redis = uniCloud.redis() exports.main = async (event, context) => { const getResult = await redis.get('my-key') const setResult = await redis.set('my-key', 'value-test') return { getResult, setResult } }; ``` **注意** - redis中,以冒号分割key,在redis的uniCloud web控制台的可视化界面中,将以tree的方式显示。折叠所有使用同一前缀的key。 比如2个key,`uni:aa`和`uni:bb`,将显示为根节点为uni的tree,展开后有aa和bb。 - 以`uni:`、`dcloud:`、`unicloud:`为前缀的redis的key,为**官方前缀**。开发者自己的业务所需的key应避免使用这些前缀。 - 调用`uniCloud.redis()`时返回的redis实例对应着一个连接,多次调用时如果存在未断开连接的redis实例则返回此实例。如果不存在redis实例或之前的redis实例已断开连接则返回新的redis实例。 - redis实例创建时并未建立与redis的连接,而是在第一次调用redis方法时才会与redis建立连接。在实际业务中的表现就是一个云函数实例第一次调用redis方法会慢上几毫秒 - 为云函数开启redis扩展会影响云函数固定ip功能,详情参考:[云函数固定出口IP](uniCloud/cf-functions.md?id=eip) ## Redis本地运行@local-function > HBuilderX 3.4.10 起支持 因为Redis在云函数的内网中,所以只能在云端云函数中访问,而不能在本地云函数中访问。每次调试Redis相关功能需要不断的上传云函数,严重影响开发效率。自HBuilderX 3.4.10起,本地云函数可以通过云端内置代理访问云端Redis。如果在本地调用云端Redis的话会在云函数日志内看到`HBuilderX运行调试Redis的代理请求`字样。 ## 数据类型@data-type Redis中数据被存储为key-value形式,key均为字符串,value有以下几种类型 ### 字符串 String 字符串类型,这是最简单Redis类型。需要注意的是Redis并没有number类型,如果存入number类型的数据最终也会转为string类型。 ```js await redis.set('string-key', 1) // 设置string-key的值为字符串"1" await redis.get('string-key') // 获取string-key的值,"1" ``` ### 列表 List 列表类型,类似JavaScript中的数组,但是有区别。严格来说List是基于链表实现的,和js中数组相比一个显著的差异就是头部插入的效率。如果你测试过往一个长度百万的数组最前面插入一位的话,你会发现这个操作会耗时很久。但是List并没有这个问题,对于List来说在前后插入数据耗时是一样的。 **注意** - list为空时对应的键会被删除,即redis内不存在空List ```js await redis.lpush('list-key', 1) // 往list-key左侧添加一个元素,不存在则创建 ``` ### 哈希表 Hash Hash类型类似js里面的Object。 ```js await redis.hmset('hash-key', 'key1', 'value1', 'key2', 'value2') // 批量为hash-key添加键值,不存在则创建 await redis.hset('hash-key', 'key1', 'value1') // 为hash-key添加键值,不存在则创建 ``` ### 集合 Set 集合是String的**无序排列**,集合内的元素不可重复 ```js await redis.sadd('set-key', 'value1', 'value2') // 往集合内添加数据,不存在则创建 ``` ### 有序集合 SortedSet 有序集合和集合一样也是string类型元素的集合,且不允许重复的成员。不同的是每个元素将有一个double类型的分数(分数不一定是连续的),用于对元素进行排序 ```js await redis.zadd('sorted-set-key', 1, 'value1') // 往有序集合内添加数据并指定分数,不存在则创建 await redis.zadd('sorted-set-key', 2, 'value2') ``` ### 基数统计 HyperLogLog 一个集合(指的是 Object 的聚合,可以包含重复元素)中不重复元素的个数。例如集合 {1,3,5,1,3},它有5个元素,但它的基数为3。 ```js await redis.pfadd('key', 'element1', 'element2') ``` ### 事务 Transaction Redis 事务可以一次执行多个命令 ```js await redis.multi() // 开启事务 redis.set('key', 'value') await redis.exec() // 执行事务块内的命令 ``` ### 位图 bitMap 是一串连续的二进制数组(0和1),可以通过偏移量(offset)定位元素。BitMap通过最小的单位bit来进行0|1的设置,表示某个元素的值或者状态,时间复杂度为O(1)。 ```js await redis.setbit('key', 'offset', 'value') // 设置指定位的值 ``` ## API@api > 此处仅列举常见命令,完整命令支持请查看[redis官方文档](https://redis.io/commands) ### 字符串 String #### get 用于获取字符串类型的数据 **接口形式** ```js await redis.get(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 此接口返回获取到的数据(字符串类型),返回null表示无此键 **示例** ```js await redis.get('string-key') // '1' ``` #### set 用于设置字符串类型数据,新增、修改均可 **接口形式** 该接口有多种形式 ```js await redis.set(key: string, value: string, flag: string) await redis.set(key: string, value: string, mode: string, duration: number) await redis.set(key: string, value: string, mode: string, duration: number, flag: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |value |值 |是 | | |duration |过期时间,到期后自动删除 |否 | | |mode |标识duration的单位 |否(duration不为空时必填) |EX:单位秒,PX:单位毫秒 | |flag |区分状态进行SET |否 |NX:不存在时才设置,XX:存在时才设置 | **返回值** 此接口返回字符串类型'OK'表示操作成功,返回null表示未更新 **示例** ```js await redis.set('string-key', 1) // redis内存储为字符串"1" await redis.set('string-key', '1', 'NX') // string-key不存在时设置为1 await redis.set('string-key', '1', 'EX', 100) // string-key 100秒后过期 await redis.set('string-key', '1', 'EX', 100, 'NX') // string-key不存在时设置为1,过期时间设置为100秒 ``` #### getrange 返回 key 中字符串值的子字符串 ```js await redis.getrange(key: string, start: number, end: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |start |起始位置 |是 | (0代表第一个字符,-1代表最后1个字符,-2代表最后第2个字符)| |end |结束位置 |是 | (0代表第一个字符,-1代表最后1个字符,-2代表最后第2个字符) | **返回值** 此接口返回截取得到的子字符串表示操作成功,返回空字符串表示未匹配 **示例** ```javascript await redis.getrange('string-key', 0, 10) ``` #### setex 键存在时,设置为指定字符串并指定过期时间 **接口形式** ```js await redis.setex(key: string, seconds: number, value: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |seconds|过期时间 |是 |单位:秒 | |value |值 |是 | | **返回值** 此接口返回字符串类型'OK'表示操作成功,返回null表示未更新 **示例** ```js await redis.setex('string-key', 10, 'value') // 值设置为value,过期时间10秒 ``` #### setnx 键不存在时,设置为指定字符串 **接口形式** ```js await redis.setnx(key: string, value: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |value |值 |是 | | **返回值** 此接口返回字符串类型'OK'表示操作成功,返回null表示未更新 **示例** ```js await redis.setnx('string-key', 'value') // 键string-key不存在时将值设置为'value' ``` #### mget 批量获取指定键的值 **接口形式** ```js await redis.mget(key1: string, key2: string, ...) ``` **入参说明** 接收一个键的列表 **返回值** 此接口按传入顺序返回获取到的数据组成的数组,存在的键返回字符串类型,不存在的键返回null **示例** ```js await redis.mget('key1', 'key2') // '1' ``` #### mset 批量设置键值 **接口形式** ```js await redis.mset(key1: string, value1: string, key2: string, value2: string, ...) ``` **入参说明** 接收一个键、值的列表 **返回值** 此接口只会返回OK **示例** ```js await redis.mset('key1', '1', 'key2', '2') // 'OK' ``` #### strlen 返回 key 所储存的字符串值的长度。 **接口形式** ```json await redis.strlen(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 此接口返回字符串长度,不存在的key会返回0 **示例** ```js await redis.strlen('key') ``` #### msetnx 批量设置键值,当且仅当所有给定 key 都不存在。 **接口形式** ```json await redis.msetnx(key1: string, value1: string, key2: string, value2: string); ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 此接口返回1表示执行成功 **示例** ```js await redis.msetnx('key1', 'value1', 'key2', 'value2') ``` #### psetex 设置key对应字符串value,并且设置key在给定的milliseconds(毫秒)时间之后超时过期 **接口形式** ```json await redis.psetex(key: string, milliseconds: number, value: string); ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |milliseconds |毫秒 |是 | | |value |值 |是 | | **返回值** 此接口返回OK表示执行成功 **示例** ```js await redis.psetex('key', 1666582638076, 'value') ``` #### incr 对指定的键执行加1操作 **接口形式** ```js await redis.incr(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 接口返回执行加一操作后的值(number类型) **注意** 操作的值并非整数形式(例:字符串"1"是整数形式,字符串"a"非整数形式)时会直接抛出错误 **示例** ```js await redis.set('string-key', '1') await redis.incr('string-key') // 2 ``` #### incrby 在指定的键上加一个整数 **接口形式** ```js await redis.incrby(key: string, increment: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |increment|增加的值 |是 | | **返回值** 接口返回执行加操作后的值(number类型) **注意** 操作的值并非整数形式(例:字符串"1"是整数形式,字符串"a"非整数形式)时会直接抛出错误 **示例** ```js await redis.set('string-key', '1') await redis.incrby('string-key', 2) // 3 ``` #### incrbyfloat 在指定的键上加一个浮点数 **接口形式** ```js await redis.incrbyfloat(key: string, increment: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |increment|增加的值,允许为负值来实现相减功能 |是 | | **返回值** 接口返回执行加操作后的值(number类型) **注意** - 操作的值并非整数形式(例:字符串"1"是整数形式,字符串"a"非整数形式)时会直接抛出错误 - 浮点数相加和js内表现一致,可能与预期结果不一致,见下方示例 **示例** ```js await redis.set('string-key', '1.1') await redis.incrbyfloat('string-key', 2.2) // 3.30000000000000027 // js内执行 0.1 + 0.2 会得到类似的值 3.3000000000000003 ``` #### decr 对指定的键执行减1操作 **接口形式** ```js await redis.decr(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 接口返回执行减1操作后的值(number类型) **注意** 操作的值并非整数形式(例:字符串"1"是整数形式,字符串"a"非整数形式)时会直接抛出错误 **示例** ```js await redis.set('string-key', '1') await redis.decr('string-key') // 0 ``` #### decrby 在指定的键上减一个整数 **接口形式** ```js await redis.decrby(key: string, decrement: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |decrement|减少的值 |是 | | **返回值** 接口返回执行加一操作后的值(number类型) **注意** 操作的值并非整数形式(例:字符串"1"是整数形式,字符串"a"非整数形式)时会直接抛出错误 **示例** ```js await redis.set('string-key', '1') await redis.decrby('string-key', 2) // -1 ``` #### append 如果 key 已经存在并且是一个字符串, APPEND 命令将指定的 value 追加到该 key 原来值(value)的末尾。 **接口形式** ```json await redis.decrby(key: string, value: string); ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |value |需要追加的值 |是 | | **返回值** 此接口返回执行成功返回值的长度 **示例** ```js await redis.decrby('key', 10); ``` ### 键 Key #### del 用于删除执行的键 **接口形式** ```js await redis.del(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 接口返回数字1表示删除成功,数字0表示键不存在删除失败 **示例** ```js await redis.del('string-key') // '1' ``` #### dump 用于序列化给定 key ,并返回被序列化的值。 **接口形式** ```js await redis.dump(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** key不存在返回null,存在返回序列化后的值 **示例** ```js await redis.set('key', 'hello') await redis.dump('key') // '\u0000\u0005hello\t\u0000����1�C�' ``` #### exists 判断一个键是否存在 **接口形式** ```js await redis.exists(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 如果key存在返回数字1,如果key不存在返回数字0 **示例** ```js await redis.exists('string-key') // 0 | 1 ``` #### expire 为指定的key设置过期时间 **接口形式** ```js await redis.expire(key: string, seconds: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |seconds|过期时间 |是 |单位:秒 | **返回值** 如果成功设置过期时间返回数字1,如果未成功存在返回数字0 **示例** ```js await redis.expire('key', 600) // 设置key为600秒后过期 ``` #### expireat EXPIREAT 的作用和 EXPIRE 类似,都用于为 key 设置过期时间。 不同在于 EXPIREAT 命令接受的时间参数是 UNIX 时间戳(unix timestamp)。 **接口形式** ```js await redis.expireat(key: string, timestamp: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |seconds|UNIX 时间戳 |是 |单位:秒 | **返回值** 如果成功设置过期时间返回数字1,如果未成功存在返回数字0 **示例** ```js await redis.expireat('key', 1666584328) // 设置key将在2022-10-24 12:05过期 ``` #### pexpire 这个命令和 expire 命令的作用类似,但是它以毫秒为单位设置 key 的生存时间,而不像 expire 命令那样,以秒为单位。 **接口形式** ```js await redis.pexpire(key: string, milliseconds: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |milliseconds|毫秒时间戳 |是 | | **返回值** 如果成功设置过期时间返回数字1,如果未成功存在返回数字0 **示例** ```js await redis.pexpire('key', 1500) // 设置key为1500毫秒后过期 ``` #### pexpireat 设置 key 过期时间的时间戳(unix timestamp) 以毫秒计 **接口形式** ```js await redis.pexpireat(key: string, timeatamp: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |timeatamp|毫秒时间戳 |是 | | **返回值** 如果成功设置过期时间返回数字1,如果未成功存在返回数字0 **示例** ```js await redis.pexpireat('key', 1666584328000) // 设置key将在2022-10-24 12:05过期 ``` #### keys 查找所有符合给定模式 pattern 的 key 。 **接口形式** ```js await redis.keys(patten: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |patten|表达式 |是 | | **返回值** 符合给定模式的 key 列表 (类型为Array)。 **示例** ```js await redis.mset('one', 1, 'two', 2, 'three', 3, 'four', 4) await redis.keys('*o*') // ['one', 'two', 'four'] await redis.keys('*') // ['one', 'two', 'three', 'four'] ``` #### persist 移除 key 的过期时间,key 将持久保持。 **接口形式** ```js await redis.persist(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key|键 |是 | | **返回值** 移除过期时间成功返回1, key不存在或者key没有设置过期时间返回0 **示例** ```js await redis.set('key', 'ok') await redis.expire('key', 10) // 10秒后过期 await redis.persist('key') // 移除 key 过期时间 ``` #### pttl 获取过期时间剩余多少毫秒 **接口形式** ```js await redis.pttl(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 如果没有设置过期时间(永久有效)返回数字-1,如果不存在或者已过期返回数字-2,否则返回剩余秒数 **示例** ```js await redis.pttl('key') ``` #### ttl 获取过期时间剩余多少秒 **接口形式** ```js await redis.ttl(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 如果没有设置过期时间(永久有效)返回数字-1,如果不存在或者已过期返回数字-2,否则返回剩余秒数 **示例** ```js await redis.ttl('key') ``` #### randonkey 从当前数据库中随机返回一个 key 。 **接口形式** ```js await redis.randonkey() ``` **入参说明** 无 **返回值** 数据库不为空时返回一个 key, 数据库为空时返回 null **示例** ```js await redis.randonkey() ``` #### rename 将 key 改名为 newkey。 当 key 和 newkey 相同,或者 key 不存在时,返回一个错误。 当 newkey 已经存在时, rename 命令将覆盖旧值。 **接口形式** ```js await redis.rename(key: string, newKey: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |newKey |新的键名 |是 | | **返回值** 修改成功返回 OK,key 不存在时返回一个错误 **示例** ```js await redis.rename('key', 'newKey') ``` #### renamenx 仅当 newkey 不存在时,将 key 改名为 newkey 。 当 key 不存在时,返回一个错误。 **接口形式** ```js await redis.renamenx(key: string, newKey: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |newKey |新的键名 |是 | | **返回值** 修改成功返回 1, newKey 存在返回 0 **示例** ```js await redis.renamenx('key', 'newKey') ``` #### scan 迭代当前数据库中的数据库键。 **接口形式** ```js await redis.scan(cursor: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |cursor |游标 |是 | | **返回值** 返回包含两个元素的数据 [下次迭代的新游标,0代表已结束, [所有被迭代的数据]] **示例** ```js await redis.mset('key1', 1, 'key2', '2', 'key3', 3, 'key4', 4, 'key5', 5) await redis.scan(0) // 返回 [6, ['key1', 'key2', 'key3', 'key4', 'key5']] ``` #### type 返回 key 所储存的值的类型。 **接口形式** ```js await redis.type(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** - none (key不存在) - string (字符串) - list (列表) - set (集合) - zset (有序集) - hash (哈希表) **示例** ```js await redis.set('key1', 1) await redis.type('key1') // 返回 string ``` ### 列表 List #### rpush 在List类型数据结尾追加数据 **接口形式** ```js await redis.rpush(key: string, value: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |value|追加的值 |是 | | **返回值** 接口返回执行追加操作后List的长度 **注意** - 如果操作的数据类型不为List,则会抛出错误 - 如果指定的key不存在,则创建一个新的List并将value追加进去 #### rpushx 用法同`rpush`,仅在list存在时才在List结尾追加数据 #### rpoplpush 移除列表的最后一个元素,并将该元素添加到另一个列表并返回。 **接口形式** ```js await redis.rpoplpush(source: string, destination: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |source |键 |是 | | |destination |另一个键 |是 | | **返回值** source存在返回被移除的元素,source 不存在返回 null **示例** ```js await redis.lrange('source', 0, -1) // 返回 ['a', 'b', 'c'] await redis.lrange('destination', 0, -1) // 返回 [] await redis.rpoplpush('source', 'destination') // 返回 c await redis.lrange('source', 0, -1) // 返回 ['a', 'b'] await redis.lrange('destination', 0, -1) // 返回 ['c'] ``` #### rpop 从List类型数据结尾删除一条数据,并返回删除的值 **注意:redis内List长度为0时会被自动删除** **接口形式** ```js await redis.rpop(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 接口返回此次操作删除的值,如果key不存在则返回null **注意** - 如果操作的数据类型不为List,则会抛出错误 #### lpush 在List类型数据开头追加数据 **接口形式** ```js await redis.lpush(key: string, value: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |value|追加的值 |是 | | **返回值** 接口返回执行追加操作后List的长度 **注意** - 如果操作的数据类型不为List,则会抛出错误 - 如果指定的key不存在,则创建一个新的List并将value追加进去 #### lpushx 用法同`lpush`,仅在list存在时才在List开头追加数据 #### lpop 从List类型数据开头删除一条数据,并返回删除的值 **注意:redis内List长度为0时会被自动删除** **接口形式** ```js await redis.rpop(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 接口返回此次操作删除的值,如果key不存在则返回null **注意** - 如果操作的数据类型不为List,则会抛出错误 #### linsert 在List内指定元素位置前或后插入元素,未匹配到指定元素时不插入 **接口形式** ```js await redis.linsert(key: string, dir: 'BEFORE' | 'AFTER', pivot: string, value: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |dir |指定在前还是后插入 |是 | | |pivot|指定要查找的元素 |是 | | |value|指定要插入的值 |是 | | **返回值** 接口返回插入后的list长度,未匹配到要查找的值时返回-1,key不存在时此接口返回0 **注意** - 如果操作的数据类型不为List,则会抛出错误 #### lindex 获取List内指定下标的元素 **接口形式** ```js await redis.lindex(key: string, index: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |index|指定下标 |是 | | **返回值** 接口返回指定下标在list内对应的值,如果key不存在则返回null **注意** - 如果操作的数据类型不为List,则会抛出错误 #### llen 返回List的长度 **接口形式** ```js await redis.llen(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 接口返回list的长度,如果key不存在则返回0 **注意** - 如果操作的数据类型不为List,则会抛出错误 #### lrange 返回列表 key 中指定区间内的元素,区间以偏移量 start 和 stop 指定。 **接口形式** ```js await redis.lrange(key: string, strar: number, end: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |start |起始下标 |是 | | |end |结束下标 |是 | | **返回值** 返回指定确定的元素,类型为Array **示例** ```js await redis.rpush('key', 'a') await redis.rpush('key', 'b') await redis.rpush('key', 'c') await redis.lrange('key', 0, 1) // 返回 ['a', 'b'] ``` #### lrem 根据参数 count 的值,移除列表中与参数 value 相等的元素。 **接口形式** ```js await redis.lrem(key: string, count: number, value: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |count |移除的数量 |是 | | |value |值 |是 | | **count说明** count 的值可以是以下几种: - count > 0 : 从表头开始向表尾搜索,移除与 value 相等的元素,数量为 count 。 - count < 0 : 从表尾开始向表头搜索,移除与 value 相等的元素,数量为 count 的绝对值。 - count = 0 : 移除表中所有与 value 相等的值。 **返回值** key 存在时返回被移除的元素的数量,key 不存在时返回 0 **示例** ```js await redis.rpush('key', 'a') await redis.lrem('key', 0, 'a') // 返回 1 ``` #### lset 将列表 key 下标为 index 的元素的值设置为 value 。 **接口形式** ```js await redis.lset(key: string, index: number, value: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |index |列表值的下标 |是 | | |value |值 |是 | | **返回值** key 存在时成功返回 OK,key 不存在时返回错误 **示例** ```js await redis.rpush('key', 'a') await redis.lset('key', 0, 'b') // 返回 OK ``` #### ltrim 修剪(trim)一个已存在的 list,这样 list 就会只包含指定范围的指定元素 **接口形式** ```js await redis.ltrim(key: string, start: number, end: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |start |起始位置 |是 | | |end |结束位置 |是 | | **注意** 超出范围的下标 超出范围的下标值不会引起错误。 如果 start 下标比列表的最大下标 end ( llen list 减去 1 )还要大,或者 start > stop , ltrim 返回一个空列表(因为 ltrim 已经将整个列表清空)。 如果 stop 下标比 end 下标还要大,Redis将 stop 的值设置为 end 。 **返回值** 执行成功返回 OK **示例** ```js await redis.ltrim('key', 0, -1) // 返回 OK ``` #### blpop 移出并获取列表的第一个元素, 如果列表没有元素会阻塞列表直到等待超时或发现可弹出元素为止。 如果列表为空,返回一个 nil 。 否则,返回一个含有两个元素的列表,第一个元素是被弹出元素所属的 key ,第二个元素是被弹出元素的值。 **接口形式** ```js await redis.blpop(key: string, timeout: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |timeout |等待超时时间;单位秒 |是 | | **返回值** 列表为空返回 null,否则返回含有两个元素的数组[键名, 值] **示例** ```js await redis.rpush('key', 'a') await redis.blpop('key', 10) // 返回 ['key', 'a'] ``` #### brpop 移出并获取列表的最后一个元素, 如果列表没有元素会阻塞列表直到等待超时或发现可弹出元素为止。 假如在指定时间内没有任何元素被弹出,则返回一个 nil 和等待时长。 反之,返回一个含有两个元素的列表,第一个元素是被弹出元素所属的 key ,第二个元素是被弹出元素的值。 **接口形式** ```js await redis.brpop(key: string, timeout: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |timeout |等待超时时间;单位秒 |是 | | **返回值** 在指定时间内没有任何元素被弹出,则返回 null 和等待时长,否则返回含有两个元素的数组[键名, 值] **示例** ```js await redis.rpush('key', 'a') await redis.brpop('key', 10) // 返回 ['key', 'a'] ``` #### brpoplpush 从 source 中取出最后一个元素,并插入到 destination 头部;如果列表没有元素会阻塞列表直到等待超时或发现可弹出元素为止。 **接口形式** ```js await redis.brpoplpush(source: string, destination: string, timeout: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |source |键 |是 | | |destination |另一个键 |是 | | |timeout |等待超时时间 |是 | | **返回值** 在指定时间内没有任何元素被弹出,则返回 null 和等待时长,否则返回含有两个元素的数组[键名, 值] **示例** ```js await redis.brpoplpush('source', 'destination', 10) ``` ### 哈希表 Hash #### hset 设置 key 中指定 field 的值。 **接口形式** ```js await redis.hset(key: string, field: string, value: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |field |字段名 |是 | | |value |值 |是 | | **注意** 如果 key 不存在,将会创建一个新的哈希表并进行 hset 操作。 如果 field 已经存在于哈希表中,旧值将被覆盖。 **返回值** 新创建的 field 返回 1, 覆盖旧值返回 0 **示例** ```js await redis.hset('key', 'field', 'value') ``` #### hget 返回 key 中指定 field 的值。 **接口形式** ```js await redis.hget(key: string, field: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |field |字段名 |是 | | **返回值** field 不存在时返回 null, 存在返回 field 值 **示例** ```js await redis.hset('key', 'field', 'value') await redis.hget('key', 'field') // 返回 value ``` #### hmset 同时在一个 key 中设置多个 field-value **接口形式** ```js await redis.hmset(key: string, field1: string, value1: string, field2: string, value2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |field1 |字段名 |是 | | |value1 |值 |是 | | **返回值** key 不是hash类型时返回错误,执行成功返回 OK **示例** ```js await redis.hmset('key', 'field1', 'value1', 'field2', 'value2') ``` #### hmget 返回一个 key 中的多个 field **接口形式** ```js await redis.hmget(key: string, field1: string, field2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |field1 |字段名 |是 | | **返回值** 返回 获取的字段列表,类型Array **示例** ```js await redis.hmget('key', 'field1', 'field2') ``` #### hgetall 返回 key 的所有 field 与 value **接口形式** ```js await redis.hgetall(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 返回 field与value,类型Array **示例** ```js await redis.hset('key', 'field1', 'value1') await redis.hset('key', 'field2', 'value2') await redis.hgetall('key') // 返回 ['field1', 'value1', 'field2', 'value2'] ``` #### hdel 删除 key 中指定的一个或多个 field **接口形式** ```js await redis.hdel(key: string, field1: string) await redis.hdel(key: string, field1: string, field2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |field1 |字段 |是 | | **返回值** 返回被删除字段的数量 **示例** ```js await redis.hdel('key') // 返回 0 ``` #### hexists 查看 key 中 field 是否存在 **接口形式** ```js await redis.hexists(key: string, field: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |field |字段 |是 | | **返回值** 字段存在返回 1, key 不存在或者字段不存在返回 0 **示例** ```js await redis.hexists('key', 'field') // 返回 0 ``` #### hincrby 为 key 中指定的 field 的值加上整数增量 increment **接口形式** ```js await redis.hincrby(key: string, field: string, increment: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |field |字段 |是 | | |increment |增量整数;>0 增加,<0 减少 |是 | | **返回值** 返回 增量增加后的值 **示例** ```js await redis.hincrby('key', 'field', 10) // 返回 10 ``` #### hincrbyfloat 为 key 中指定的 field 的值加上浮点数增量 increment **接口形式** ```js await redis.hincrbyfloat(key: string, field: string, increment: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |field |字段 |是 | | |increment |增量浮点数;>0 增加,<0 减少 |是 | | **返回值** 返回 增量增加后的值 **示例** ```js await redis.hincrbyfloat('key', 'field', 2.5) // 返回 2.5 ``` #### hkeys 返回 key 中所有 field **接口形式** ```js await redis.hkeys(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 返回所有 field;类型Array,key 不存在时返回 空数组 **示例** ```js await redis.hkeys('key') // 返回 [] ``` #### hlen 返回 key 中 field 数量 **接口形式** ```js await redis.hlen(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 返回 field 数量,key 不存在时返回 0 **示例** ```js await redis.hlen('key') // 返回 0 ``` #### hsetnx 当 key 中 field 不存在时设置 value,当 field 已存在,该操作无效。 **接口形式** ```js await redis.hsetnx(key: string, field: string, value: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |field |字段 |是 | | |value |值 |是 | | **返回值** 设置成功返回 1, field 已存在返回 0 **示例** ```js await redis.hsetnx('key', 'field', 'value') // 返回 1 ``` #### hvals 返回 key 中所有 field 的 value **接口形式** ```js await redis.hvals(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 返回所有 field 的 value,类型Array,key 不存在时返回空数组 **示例** ```js await redis.hvals('key') // 返回 [] ``` #### hscan 迭代哈希键中的键值对 **接口形式** ```js await redis.hscan(key: string, cursor: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |cursor |游标 |是 | | **返回值** 返回包含两个元素的数据 [下次迭代的新游标,0代表已结束, [所有被迭代的数据]] **示例** ```js await redis.hscan('key', 0) // 返回 [] ``` ### 集合 Set #### sadd 将一个或多个 member 加入到集合 key 中,已经在集合中的 member 元素将忽略。 **接口形式** ```js await redis.sadd(key: string, member: string) await redis.sadd(key: string, member1: string, member2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | |member |成员 |是 | | **返回值** 返回被添加到集合中的新元素的数量,不包括被忽略的元素。 **示例** ```js await redis.sadd('key', 'm1', 'm2') // 返回 2 ``` #### scard 返回集合存储的key的基数(集合中元素的数量) **接口形式** ```js await redis.scard(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |键 |是 | | **返回值** 返回集合存储的key的基数,key 不存在时返回 0 **示例** ```js await redis.sadd('key', 'a', 'b') await redis.scard('key') // 返回 2 ``` #### sdiff 返回一个集合的全部成员,该集合是所有给定集合之间的差集。 不存在的 key 被视为空集。 **接口形式** ```js await redis.sdiff(key1: string, key2: string) await redis.sdiff(key1: string, key2: string, key3: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key1 |集合的键 |是 | | |key2 |集合的键 |是 | | **返回值** 返回差集成员的列表。类型Array **示例** ```js await redis.sadd('key1', 'a', 'b', 'c') await redis.sadd('key2', 'c', 'd', 'f') await redis.sdiff('key1', 'key2') // 返回 ['b', 'a'] ``` #### sdiffstore 这个命令的作用和 sdiff 类似,但它将结果保存到 destination 集合,而不是简单地返回结果集。 如果 destination 集合已经存在,则将其覆盖。 destination 可以是 key 本身。 **接口形式** ```js await redis.sdiffstore(destination: string, key1: string, key2: string) await redis.sdiffstore(destination: string, key1: string, key2: string, key3: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |destination |需要存储的键名 |是 | | |key1 |集合的键 |是 | | |key2 |集合的键 |是 | | **返回值** 返回结果集中的元素数量 **示例** ```js await redis.sadd('key1', 'a', 'b', 'c') await redis.sadd('key2', 'c', 'd', 'f') await redis.sdiff('destination', 'key1', 'key2') // 返回 2 ``` #### sinter 返回一个集合的全部成员,该集合是所有给定集合的交集。 不存在的 key 被视为空集。 当给定集合当中有一个空集时,结果也为空集(根据集合运算定律)。 **接口形式** ```js await redis.sinter(key1: string, key2: string) await redis.sinter(key1: string, key2: string, key3: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key1 |集合的键 |是 | | |key2 |集合的键 |是 | | **返回值** 返回交集成员的列表,类型为Array **示例** ```js await redis.sadd('key1', 'a', 'b', 'c') await redis.sadd('key2', 'c', 'd', 'f') await redis.sinter('key1', 'key2') // 返回 ['c'] ``` #### sinterstore 这个命令类似于 sinter 命令,但它将结果保存到 destination 集合,而不是简单地返回结果集。 如果 destination 集合已经存在,则将其覆盖。 destination 可以是 key 本身。 **接口形式** ```js await redis.sinterstore(destination: string, key1: string, key2: string) await redis.sinterstore(destination: string, key1: string, key2: string, key3: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |destination |需要存储的键名 |是 | | |key1 |集合的键 |是 | | |key2 |集合的键 |是 | | **返回值** 返回结果集中的成员数量。 **示例** ```js await redis.sadd('key1', 'a', 'b', 'c') await redis.sadd('key2', 'c', 'd', 'f') await redis.sinterstore('destination', 'key1', 'key2') // 返回 1 ``` #### sismember 判断 member 元素是否是集合 key 成员。 **接口形式** ```js await redis.sismember(key: string, member: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | |member |成员 |是 | | **返回值** member 元素是集合的成员返回 1,不是集合成员返回或者 key 不存在返回 0 **示例** ```js await redis.sadd('key', 'a', 'b', 'c') await redis.sismember('key', 'a') // 返回 1 ``` #### smembers 返回集合 key 中的所有成员。 **接口形式** ```js await redis.smembers(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | **返回值** 返回集合中的所有成员,类型Array **示例** ```js await redis.sadd('key', 'a', 'b', 'c') await redis.smembers('key', 'a') // 返回 ['a', 'b', 'c'] ``` #### smove 将 member 元素从 source 集合移动到 destination 集合。 **接口形式** ```js await redis.smove(source: string, destination: string, member: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |source |集合的键 |是 | | |destination |集合的键 |是 | | |member |要移动的成员 |是 | | **返回值** member 移除成功返回 1,member 不是 source 集合成员,并且没有任何操作对 destination 集合执行,那么返回 0 。 **示例** ```js await redis.sadd('source', 'a', 'b', 'c') await redis.smove('source', 'destination', 'a') // 返回 1 ``` #### spop 移除并返回集合中的一个随机元素。 **接口形式** ```js await redis.spop(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | **返回值** 返回被移除的随机元素。key 不存在或 key 是空集时,返回 null 。 **示例** ```js await redis.spop('key') ``` #### srandmember 在集合 key 中随机获取一个或者 count 个元素 **接口形式** ```js await redis.srandmember(key: string) await redis.srandmember(key: string, count: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | |count |返回几个元素 |否 | | **count说明** - 如果 count 为正数,且小于集合基数,那么命令返回一个包含 count 个元素的数组,数组中的元素各不相同。如果 count 大于等于集合基数,那么返回整个集合。 - 如果 count 为负数,那么命令返回一个数组,数组中的元素可能会重复出现多次,而数组的长度为 count 的绝对值。 **返回值** 只提供 key 参数时,返回一个元素;如果集合为空,返回 nil 。 如果提供了 count 参数,那么返回一个数组;如果集合为空,返回空数组。 **示例** ```js await redis.srandmember('key') ``` #### srem 移除集合 key 中的一个或多个 member 元素,不存在的 member 元素会被忽略。 **接口形式** ```js await redis.srem(key: string, member: string) await redis.srem(key: string, member1: string, member2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | |member |成员 |是 | | **返回值** 返回被成功移除的元素的数量,不包括被忽略的元素。 **示例** ```js await redis.srem('key', 'member') ``` #### sunion 返回一个集合的全部成员,该集合是所有给定集合的并集。 **接口形式** ```js await redis.sunion(key: string) await redis.sunion(key1: string, key2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | |member |成员 |是 | | **返回值** 返回并集成员的列表。 **示例** ```js await redis.sunion('key1', 'key2') ``` #### sunionstore 这个命令类似于 sunion 命令,但它将结果保存到 destination 集合,而不是简单地返回结果集。 如果 destination 已经存在,则将其覆盖。 destination 可以是 key 本身。 **接口形式** ```js await redis.sunionstore(destination: string, key: string) await redis.sunionstore(destination: string, key1: string, key2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |destination |集合的键 |是 | | |key |集合的键 |是 | | **返回值** 返回结果集中的元素数量。 **示例** ```js await redis.sunionstore('destination', 'key') ``` #### sscan 迭代集合键中的元素。 **接口形式** ```js await redis.sscan(key: string, cursor: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | |cursor |游标 |是 | | **返回值** 返回包含两个元素的数据 [下次迭代的新游标,0代表已结束, [所有被迭代的数据]] **示例** ```js await redis.sscan('key', 0) ``` ### 有序集合 SortedSet #### zadd 将一个或多个 member 元素及其 score 值加入到有序集 key 当中。 如果某个 member 已经是有序集的成员,那么更新这个 member 的 score 值,并通过重新插入这个 member 元素,来保证该 member 在正确的位置上。 **接口形式** ```js await redis.zadd(key: string, score: string, member: string) await redis.zadd(key: string, score1: string, member1: string, score2: string, member2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | |score |分数; 可以是整数值或双精度浮点数 |是 | | |member |元素 |是 | | **返回值** 返回被成功添加的新成员的数量,不包括那些被更新的、已经存在的成员。 **示例** ```js await redis.zadd('key', '0', 'a') ``` #### zcard 返回有序集 key 的基数。 **接口形式** ```js await redis.zcard(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | **返回值** 当 key 存在且是有序集类型时,返回有序集的基数。 当 key 不存在时,返回 0 。 **示例** ```js await redis.zcard('key') ``` #### zcount 返回有序集 key 中, score 值在 min 和 max 之间(默认包括 score 值等于 min 或 max )的成员的数量。 **接口形式** ```js await redis.zcount(key: string, min: number, max: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | |min |最小分数 |是 | | |max |最大分数 |是 | | **返回值** 返回 score 值在 min 和 max 之间的成员的数量。 **示例** ```js await redis.zcount('key', 0, -1) ``` #### zincrby 为有序集 key 的成员 member 的 score 值加上增量 increment 。 **接口形式** ```js await redis.zincrby(key: string, increment: number, member: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合的键 |是 | | |increment |增量分值 |是 | | |member |成员 |是 | | **返回值** 返回 member 成员的新 score 值,以字符串形式表示。 **示例** ```js await redis.zincrby('key', 10, 'member') ``` #### zinterstore 计算给定的一个或多个有序集的交集,其中给定 key 的数量必须以 numkeys 参数指定,并将该交集(结果集)储存到 destination 。 **接口形式** ```js await redis.zinterstore(destination: string, numkeys: number, key: string) await redis.zinterstore(destination: string, numkeys: number, key1: string, key2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |destination |集合的键 |是 | | |numkeys |集合数量 |是 | | |key |集合键名 |是 | | **返回值** 返回保存到 destination 的结果集的基数。 **示例** ```js await redis.zinterstore('destination', 1, 'key') ``` #### zrange 返回有序集 key 中,指定区间内的成员。 **接口形式** ```js await redis.zrange(key: string, start: number, stop: number) await redis.zrange(key: string, start: number, stop: number, 'WITHSCORES') ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合键名 |是 | | |start |起始位置 |是 | | |stop |结束位置 |是 | | **返回值** 返回指定区间内,带有 score 值(可选)的有序集成员的列表,类型Array **示例** ```js await redis.zrange('key', 0, -1) ``` #### zrank 返回有序集 key 中成员 member 的排名。其中有序集成员按 score 值递增(从小到大)顺序排列。 **接口形式** ```js await redis.zrank(key: string, member: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合键名 |是 | | |member |成员 |是 | | **返回值** 如果 member 是有序集 key 的成员,返回 member 的排名。 如果 member 不是有序集 key 的成员,返回 null 。 **示例** ```js await redis.zrank('key', 'member') ``` #### zrem 移除有序集 key 中的一个或多个成员,不存在的成员将被忽略。 **接口形式** ```js await redis.zrem(key: string, member: string) await redis.zrem(key: string, member1: string, member2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合键名 |是 | | |member |成员 |是 | | **返回值** 返回被成功移除的成员的数量,不包括被忽略的成员。 **示例** ```js await redis.zrem('key', 'member') ``` #### zrevrange 返回有序集 key 中,指定区间内的成员,其中成员的位置按 score 值递减(从大到小)来排列。 **接口形式** ```js await redis.zrevrange(key: string, start: number, stop: number) await redis.zrevrange(key: string, start: number, stop: number, 'WITHSCORES') ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合键名 |是 | | |start |起始位置 |是 | | |stop |结束位置 |是 | | **返回值** 返回指定区间内,带有 score 值(可选)的有序集成员的列表,类型Array。 **示例** ```js await redis.zrevrange('key', 0,-1) ``` #### zrevrangebyscore 通过字典区间返回有序集合的成员(具有分页功能)(集合按照分值从小到大排序) **接口形式** ```js await redis.zrevrangebyscore(key: string, min: number, max: number) await redis.zrevrangebyscore(key: string, min: number, max: number, withscores: string, limit: string, offset: number, count: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合键名 |是 | | |min |字典中排序位置较小的成员,必须以”[“开头,或者以”(“开头,可使用”-“代替 |是 | | |max |字典中排序位置较大的成员,必须以”[“开头,或者以”(“开头,可使用”+”代替 |是 | | |withscores |WITHSCORES |否 | | |limit |返回结果是否分页,指令中包含LIMIT后offset、count必须输入 |否 | | |offset |返回结果起始位置 |否 | | |count |返回结果数量 |否 | | **返回值** 返回指定区间内,带有 score 值(可选)的有序集成员的列表。 **示例** ```js await redis.zrevrangebyscore('key', 1000,200) ``` #### zrevrank 返回有序集key中成员member的排名,其中有序集成员按score值从高到低排列。 排名从0开始,也就是说,score值最大的成员排名为0。 **接口形式** ```js await redis.zrevrank(key: string, member: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合键名 |是 | | |member |成员 |是 | | **返回值** 如果 member 是有序集 key 的成员,返回 member 的排名。 如果 member 不是有序集 key 的成员,返回 null 。 **示例** ```js await redis.zrevrank('key', 'member') ``` #### zscore 获取有序集 key 中,成员 member 的 score 值。 **接口形式** ```js await redis.zscore(key: string, member: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合键名 |是 | | |member |成员 |是 | | **返回值** 返回member 成员的 score 值,以字符串形式表示,key 不存在或者 member 不是 key 的成员,返回 null 。 **示例** ```js await redis.zscore('key', 'member') ``` #### zunionstore 计算给定的一个或多个有序集的并集,其中给定 key 的数量必须以 numkeys 参数指定,并将该并集(结果集)储存到 destination 。 **接口形式** ```js await redis.zunionstore(destination: string, numkeys: string, key: string) await redis.zunionstore(destination: string, numkeys: string, key1: string, key2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |destination |集合键名 |是 | | |numkeys |集合数量 |是 | | |key |集合键名 |是 | | **返回值** 返回保存到 destination 的结果集的基数。 **示例** ```js await redis.zunionstore('destination', 1, 'key') ``` #### zscan 迭代有序集合中的元素(包括元素成员和元素分值) **接口形式** ```js await redis.zscan(key: string, cursor: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |-- | |key |集合键名 |是 | | |cursor |游标 |是 | | **返回值** 返回包含两个元素的数据 [下次迭代的新游标,0代表已结束, [所有被迭代的数据]] **示例** ```js await redis.zscan('key', 0) ``` ### 基数统计 HyperLogLog #### pfadd 添加指定元素到 HyperLogLog 中。 **接口形式** ```js await redis.pfadd(key: string, element1: string, element2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |--| |key |键名 |是 | | |element1 |元素1 |是 | | |element2 |元素2 |是 | | **返回值** 如果至少有个元素被添加返回 1, 否则返回 0。 **示例** ```js await redis.pfadd('key', 'element1', 'element2') ``` #### pfcount 返回给定 HyperLogLog 的基数估算值。 **接口形式** ```js await redis.pfcount(key: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |--| |key |键名 |是 | | **返回值** 返回给定 HyperLogLog 的基数值,如果多个 HyperLogLog 则返回基数估值之和。 **示例** ```js await redis.pfadd('key', 'a', 'b', 'c', 'b', 'c') await redis.pfcount('key') // 返回 3 ``` #### pfmerge 将多个 HyperLogLog 合并为一个 HyperLogLog **接口形式** ```js await redis.pfmerge(destkey: string, sourcekey: string) await redis.pfmerge(destkey: string, sourcekey1: string, sourceKey2: string) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |--| |destKey |目标键名 |是 | | |sourceKey |键名 |是 | | **返回值** 成功返回 OK **示例** ```js await redis.pfadd('dest', 'a', 'b', 'c', 'b', 'c') await redis.pfadd('source', 'a', 'b', 'c', 'b', 'c') await redis.pfmerge('key', 'dest', 'source') await redis.pfcount('key') // 返回 3 ``` ### 事务 Transaction #### multi 将多条指令作为一个原子执行。 **示例** ```js const multi = redis.multi() multi.set('key1', 'value1') multi.set('key2', 'value2') multi.set('key3', 'value3') multi.set('key4', 'value4') const res = await multi.exec() // 如果执行成功 res = ['OK','OK','OK','OK'] // 某个操作出现错误 res = ['OK','OK', error, 'OK'] // error为 Error对象的实例 ``` #### exec 执行所有事务块内的命令。 **接口形式** ```js await redis.exec() ``` #### discard 取消事务,放弃执行事务块内的所有命令。 **接口形式** ```js await redis.discard() ``` ### 位图 BitMap #### setbit 针对key存储的字符串值,设置或清除指定偏移量offset上的位(bit) 位的设置或清除取决于value值,即1或0 **接口形式** ```js await redis.setbit(key: string, offset: number, value: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |--| |key |键名 |是 | | |offset |位 |是 | | |value |位的值;0或1 |是 | | **返回值** 返回偏移量offset位置的原始值 **示例** ```js await redis.setbit('key', 0, 1) ``` #### getbit 返回key对应的字符串,offset 位置的位(bit) **接口形式** ```js await redis.getbit(key: string, offset: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |--| |key |键名 |是 | | |offset |位 |是 | | **返回值** 返回偏移量offset位置的bit值, key 不存在时或当offset大于值的长度时返回 0 **示例** ```js await redis.setbit('key', 1, 1) await redis.getbit('key', 1) // 返回 1 ``` #### bitcount 统计给定字符串中,bit为1的数量 **接口形式** ```js await redis.bitcount(key: string, start: number, end: number) ``` **入参说明** |参数 |说明 |必填 |说明 | |-- |-- |-- |--| |key |键名 |是 | | |start |开始位 |是 | | |end |结束位 |是 | | **返回值** 返回bit值为1的数量,如果key不存在,返回0 **示例** ```js await redis.setbit('key', 1, 1) await redis.bitcount('key', 0, -1) // 返回 1 ``` #### bitop 对一个或多个保存二进制位的字符串key进行位元操作,并将结果保存到destkey上。 - operation可以是AND、OR、NOT、XOR这四种操作中的任意一种。 - bitop AND destkey key [key ...] ,对一个或多个key求逻辑并,并将结果保存到destkey - bitop OR destkey key [key ...] ,对一个或多个key求逻辑或,并将结果保存到destkey - bitop XOR destkey key [key ...] ,对一个或多个key求逻辑异或,并将结果保存到destkey - bitop NOT destkey key ,对给定key求逻辑非,并将结果保存到destkey - 除了NOT操作外,其他操作都可以接受一个或多个key作为输入 - 当bitop处理不同长度的字符串时,较短的那个字符串所缺少的部分会被看做0 - 空的key也被看作是包含0的字符串序列 **接口形式** ```js await redis.bitop(operation: string, destkey: string, key: string) await redis.bitop(operation: string, destkey: string, key1: string, key2: string) ``` **入参说明** |参数 |说明 |必填 | |-- |-- |-- | |operation |操作; AND、OR、NOT、XOR 其中一个 |是 | |destkey |键名 |是 | |key |键名 |是 | **返回值** 返回保存到 destkey 的字符串的长度(和参数给定的key中最长的字符串长度相等) **示例** ```js await redis.bitop('AND', 'andkey', 'key1', 'key2') ``` #### bitpos 返回字符串中,从左到右,第一个比特值为bit(0或1)的偏移量 **接口形式** ```js await redis.bitpos(key: string, value: string, start: number, end: number) ``` **入参说明** |参数 |说明 |必填 | |-- |-- |-- | |key |键名 |是 | |value |值;0或1 |是 | |start |开始字节,注意1字节=8位 |是 | |end |结束字节,注意1字节=8位 |是 | **返回值** 返回第一个比特值为指定bit(0或1)的偏移量 **示例** ```js await redis.set('key', 'a') await redis.bitpos('key', 1) // 返回 1 ``` ### 执行lua脚本@eval **接口形式** ```js await redis.eval(String script, Number argsCount, String key1, String key2 , ... , String arg1, String arg2, ...) ``` **参数说明** |参数 |类型 |必填 |说明 | |-- |-- |-- |-- | |script |String |是 |lua脚本内容 | |argsCount |Number |是 |参数个数,没有参数则传0 | |key1、key2... |String |否 |从eval的第三个参数开始算起,表示在脚本中所用到的那些 Redis 键(key),这些键名参数可以在 Lua 中通过全局变量 KEYS 数组,用 1 为基址的形式访问( KEYS[1] , KEYS[2] ,以此类推) | |arg1、agr2... |Number |是 |附加参数,在 Lua 中通过全局变量 ARGV 数组访问,访问的形式和 KEYS 变量类似( ARGV[1] 、 ARGV[2] ,诸如此类) | 某些情况下需要使用复杂的原子操作以避免高并发下数据修改混乱的问题,这种需求一般可通过执行lua脚本实现。如以下示例,判断redis中不存在key-test时,将其值设置为1;存在且小于10时进行加一操作;大于等于10时不进行操作直接返回。 `{0, 1}`是lua内的table类型,返回到云函数时会转为数组对应的值为`[0, 1]` ```js const [operationType, currentValue] = await redis.eval(`local val = redis.call('get','key-test') local valNum = tonumber(val) if (val == nil) then redis.call('set', 'key-test', 1) return {0, 1} end if (valNum < 10) then redis.call('incrby', 'key-test', 1) return {1, valNum + 1} else return {2, valNum} end `, 0) ``` ### quit@quit 断开redis连接,会等待redis请求执行完成后才断开连接 **接口形式** ```js await redis.quit() ``` **入参说明** 无 **返回值** 调用成功后返回`OK`字符串 **注意** - 断开连接后使用uniCloud.redis()返回的redis实例的连接将不再可用,再下次用到redis方法时需要重新调用`uniCloud.redis()`方法建立连接 ## FAQ@faq - 云函数与redis的连接 和传统开发不同,云函数实例之间是不互通的,也就是说每个使用redis的函数实例都会和redis建立一个连接,在云函数实例复用时此连接也会复用。 ## 最佳实践 ### 高并发下抢购/秒杀/防超卖示例@snap-over-sell 可以利用redis的原子操作保证在高并发下不会超卖,以下为一个简单示例 在抢购活动开始前可以将商品库存同步到redis内,实际业务中可以通过提前访问一次抢购页面加载所有商品来实现。下面通过一个简单的演示代码来实现 ```js const redis = uniCloud.redis() const goodsList = [{ // 商品库存信息 id: 'g1', stock: 100 }, { id: 'g2', stock: 200 }, { id: 'g3', stock: 400 }] const stockKeyPrefix = 'stock_' async function init() { // 抢购活动开始前在redis初始化库存 for (let i = 0; i < goodsList.length; i++) { const { id, stock } = goodsList[i]; await redis.set(`${stockKeyPrefix}${id}`, stock) } } init() ``` 抢购的逻辑见以下代码 ```js exports.main = async function (event, context) { // 一些判断抢购活动是否开始/结束的逻辑,未开始直接返回 const cart = [{ // 购物车信息,此处演示购物车抢购的例子,如果抢购每次下单只能购买一件商品,可以适当调整代码 id: 'g1', // 商品id amount: parseInt(Math.random() * 5 + 5) // 购买数量 }, { id: 'g2', amount: parseInt(Math.random() * 5 + 5) }] /** * 使用redis的eval方法执行lua脚本判断各个商品库存能否满足购物车购买的数量 * 如果不满足,返回库存不足的商品id * 如果满足,返回一个空数组 * * eval为原子操作可以在高并发下保证不出错 **/ let checkAndSetStock = ` local cart = {${cart.map(item => `{id='${item.id}',amount=${item.amount}}`).join(',')}} local amountList = redis.call('mget',${cart.map(item => `'${stockKeyPrefix}${item.id}'`).join(',')}) local invalidGoods = {} for i,cartItem in ipairs(cart) do if (cartItem['amount'] > tonumber(amountList[i])) then table.insert(invalidGoods, cartItem['id']) break end end if(#invalidGoods > 0) then return invalidGoods end for i,cartItem in ipairs(cart) do redis.call('decrby', '${stockKeyPrefix}'..cartItem['id'], cartItem['amount']) end return invalidGoods ` const invalidGoods = await redis.eval(checkAndSetStock, 0) if (invalidGoods.length > 0) { return { code: -1, message: `以下商品库存不足:${invalidGoods.join(',')}`, // 此处为简化演示代码直接返回商品id给客户端 invalidGoods // 返回库存不足的商品列表 } } // redis库存已扣除,将库存同步到数据库。为用户创建订单 // 此处代码省略... return { code: 0, message: '下单成功,跳转订单页面' } } ```