Redis笔记_5

这篇博客介绍了Redis中Hset命令的使用,包括其语法、版本要求、返回值以及实例。同时提到了Zrevrangebyscore命令,用于按分数值递减排序获取有序集合成员,还探讨了Redis作为配置存储和锁的实现策略。



第五章

5.3 查找IP所属城市及国家

使用第三方提供的IP所属城市地区库, 如http://www.ip138.com/


使用的redis api

Redis Hset 命令用于为哈希表中的字段赋值 。

如果哈希表不存在,一个新的哈希表被创建并进行 HSET 操作。

如果字段已经存在于哈希表中,旧值将被覆盖。

语法

redis Hset 命令基本语法如下:

redis 127.0.0.1:6379> HSET KEY_NAME FIELD VALUE 

可用版本

>= 2.0.0

返回值

如果字段是哈希表中的一个新建字段,并且值设置成功,返回 1 。 如果哈希表中域字段已经存在且旧值已被新值覆盖,返回 0 。

实例

redis 127.0.0.1:6379> HSET myhash field1 "foo"
OK
redis 127.0.0.1:6379> HGET myhash field1
"foo"

redis 127.0.0.1:6379> HSET website google "www.g.cn"       # 设置一个新域
(integer) 1

redis 127.0.0.1:6379>HSET website google "www.google.com" # 覆盖一个旧域
(integer) 0


Redis Zrevrangebyscore 返回有序集中指定分数区间内的所有的成员。有序集成员按分数值递减(从大到小)的次序排列。

具有相同分数值的成员按字典序的逆序(reverse lexicographical order )排列。

除了成员按分数值递减的次序排列这一点外, ZREVRANGEBYSCORE 命令的其他方面和 ZRANGEBYSCORE 命令一样。



语法

redis zrevrangebyscore 命令基本语法如下:

redis 127.0.0.1:6379> ZREVRANGEBYSCORE key max min [WITHSCORES] [LIMIT offset count]

可用版本

>= 2.2.0

返回值

指定区间内,带有分数值(可选)的有序集成员的列表。

实例

redis 127.0.0.1:6379> ZADD salary 10086 jack
(integer) 1
redis > ZADD salary 5000 tom
(integer) 1
redis 127.0.0.1:6379> ZADD salary 7500 peter
(integer) 1
redis 127.0.0.1:6379> ZADD salary 3500 joe
(integer) 1

redis 127.0.0.1:6379> ZREVRANGEBYSCORE salary +inf -inf   # 逆序排列所有成员
1) "jack"
2) "peter"
3) "tom"
4) "joe"

redis 127.0.0.1:6379> ZREVRANGEBYSCORE salary 10000 2000  # 逆序排列薪水介于 10000 和 2000 之间的成员
1) "peter"
2) "tom"
3) "joe"

5.4 发我的发现与配置

- 通过redis来存储配置信息, 这样可以试试检查系统的运行配置情况

- 可以为每个应用配置一个Redis服务器, 一个机器可以安装多台redis服务器, 只要端口不同就可以了, 一个redis可以使用多个库,


涉及的命令:

GET key

返回 key 所关联的字符串值。

如果 key 不存在那么返回特殊值 nil 。

假如 key 储存的值不是字符串类型,返回一个错误,因为 GET 只能用于处理字符串值。

可用版本:
>= 1.0.0
时间复杂度:
O(1)
返回值:
当  key 不存在时,返回  nil ,否则,返回  key 的值。
如果  key 不是字符串类型,那么返回一个错误。
<pre name="code" class="plain"># 对不存在的 key 或字符串类型 key 进行 GET
<pre name="code" class="javascript">
redis> GET db(nil)redis> SET db redisOKredis> GET db"redis"# 对不是字符串类型的 key 进行 GETredis> DEL db(integer) 1redis> LPUSH db redis mongodb mysql(integer) 3redis> GET db(error) ERR Operation against a key holding the wrong kind of value
 

 


SET key value [EX seconds] [PX milliseconds] [NX|XX]

将字符串值 value 关联到 key 。

如果 key 已经持有其他值, SET 就覆写旧值,无视类型。

对于某个原本带有生存时间(TTL)的键来说, 当 SET 命令成功在这个键上执行时, 这个键原有的 TTL 将被清除。

可选参数

从 Redis 2.6.12 版本开始, SET 命令的行为可以通过一系列参数来修改:

  • EX second :设置键的过期时间为 second 秒。 SET key value EX second 效果等同于 SETEX key second value 。
  • PX millisecond :设置键的过期时间为 millisecond 毫秒。 SET key value PX millisecond 效果等同于 PSETEX key millisecond value
  • NX :只在键不存在时,才对键进行设置操作。 SET key value NX 效果等同于 SETNX key value 。
  • XX :只在键已经存在时,才对键进行设置操作。

因为 SET 命令可以通过参数来实现和 SETNX 、 SETEX 和 PSETEX 三个命令的效果,所以将来的 Redis 版本可能会废弃并最终移除SETNX 、 SETEX 和 PSETEX 这三个命令。

可用版本:
>= 1.0.0
时间复杂度:
O(1)
返回值:

在 Redis 2.6.12 版本以前, SET 命令总是返回 OK 。

从 Redis 2.6.12 版本开始,  SET 在设置操作成功完成时,才返回  OK 。
如果设置了  NX 或者  XX ,但因为条件没达到而造成设置操作未执行,那么命令返回空批量回复(NULL Bulk Reply)。
# 对不存在的键进行设置

redis 127.0.0.1:6379> SET key "value"
OK

redis 127.0.0.1:6379> GET key
"value"


# 对已存在的键进行设置

redis 127.0.0.1:6379> SET key "new-value"
OK

redis 127.0.0.1:6379> GET key
"new-value"


# 使用 EX 选项

redis 127.0.0.1:6379> SET key-with-expire-time "hello" EX 10086
OK

redis 127.0.0.1:6379> GET key-with-expire-time
"hello"

redis 127.0.0.1:6379> TTL key-with-expire-time
(integer) 10069


# 使用 PX 选项

redis 127.0.0.1:6379> SET key-with-pexpire-time "moto" PX 123321
OK

redis 127.0.0.1:6379> GET key-with-pexpire-time
"moto"

redis 127.0.0.1:6379> PTTL key-with-pexpire-time
(integer) 111939


# 使用 NX 选项

redis 127.0.0.1:6379> SET not-exists-key "value" NX
OK      # 键不存在,设置成功

redis 127.0.0.1:6379> GET not-exists-key
"value"

redis 127.0.0.1:6379> SET not-exists-key "new-value" NX
(nil)   # 键已经存在,设置失败

redis 127.0.0.1:6379> GEt not-exists-key
"value" # 维持原值不变


# 使用 XX 选项

redis 127.0.0.1:6379> EXISTS exists-key
(integer) 0

redis 127.0.0.1:6379> SET exists-key "value" XX
(nil)   # 因为键不存在,设置失败

redis 127.0.0.1:6379> SET exists-key "value"
OK      # 先给键设置一个值

redis 127.0.0.1:6379> SET exists-key "new-value" XX
OK      # 设置新值成功

redis 127.0.0.1:6379> GET exists-key
"new-value"


# NX 或 XX 可以和 EX 或者 PX 组合使用

redis 127.0.0.1:6379> SET key-with-expire-and-NX "hello" EX 10086 NX
OK

redis 127.0.0.1:6379> GET key-with-expire-and-NX
"hello"

redis 127.0.0.1:6379> TTL key-with-expire-and-NX
(integer) 10063

redis 127.0.0.1:6379> SET key-with-pexpire-and-XX "old value"
OK

redis 127.0.0.1:6379> SET key-with-pexpire-and-XX "new value" PX 123321
OK

redis 127.0.0.1:6379> GET key-with-pexpire-and-XX
"new value"

redis 127.0.0.1:6379> PTTL key-with-pexpire-and-XX
(integer) 112999


# EX 和 PX 可以同时出现,但后面给出的选项会覆盖前面给出的选项

redis 127.0.0.1:6379> SET key "value" EX 1000 PX 5000000
OK

redis 127.0.0.1:6379> TTL key
(integer) 4993  # 这是 PX 参数设置的值

redis 127.0.0.1:6379> SET another-key "value" PX 5000000 EX 1000
OK

redis 127.0.0.1:6379> TTL another-key
(integer) 997   # 这是 EX 参数设置的值

使用模式

命令 SET resource-name anystring NX EX max-lock-time 是一种在 Redis 中实现锁的简单方法。

客户端执行以上的命令:

  • 如果服务器返回 OK ,那么这个客户端获得锁。
  • 如果服务器返回 NIL ,那么客户端获取锁失败,可以在稍后再重试。

设置的过期时间到达之后,锁将自动释放。

可以通过以下修改,让这个锁实现更健壮:

  • 不使用固定的字符串作为键的值,而是设置一个不可猜测(non-guessable)的长随机字符串,作为口令串(token)。
  • 不使用 DEL 命令来释放锁,而是发送一个 Lua 脚本,这个脚本只在客户端传入的值和键的口令串相匹配时,才对键进行删除。

这两个改动可以防止持有过期锁的客户端误删现有锁的情况出现。

以下是一个简单的解锁脚本示例:

if redis.call("get",KEYS[1]) == ARGV[1]
then
    return redis.call("del",KEYS[1])
else
    return 0
end

这个脚本可以通过 EVAL ...script... 1 resource-name token-value 命令来调用。





### 黑马程序员 Redis 学习笔记 Redis 是一种高性能的键值存储系统,广泛应用于缓存、消息队列等领域。以下是基于提供的引用内容以及相关知识点整理的关于 Redis 的学习笔记。 #### 一、Redis 基础概念 Redis(Remote Dictionary Server)是一个开源的内存数据结构存储系统,支持多种数据类型的高效操作。它可以用作数据库、缓存和消息中间件[^4]。 主要特点包括: - 高性能:所有数据都保存在内存中,读写速度极快。 - 数据持久化:通过 RDB 和 AOF 方式实现数据持久化。 - 多种数据类型支持:字符串(String)、哈希(Hash)、列表(List)、集合(Set)、有序集合(Sorted Set)[^4]。 --- #### 二、Spring Data Redis 中的 RedisTemplate 工具类 Spring Data Redis 提供了 `RedisTemplate` 类作为与 Redis 进行交互的核心工具。该工具封装了对 Redis 各种数据类型的操作方法,并提供了灵活的配置选项[^4]。 ##### 1. 不同数据类型的操作 API | 方法 | 返回接口 | 功能描述 | |--------------------------|------------------------|-------------------------| | `opsForValue()` | ValueOperations | 操作 String 类型数据 | | `opsForHash()` | HashOperations | 操作 Hash 类型数据 | | `opsForList()` | ListOperations | 操作 List 类型数据 | | `opsForSet()` | SetOperations | 操作 Set 类型数据 | | `opsForZSet()` | ZSetOperations | 操作 SortedSet 类型数据 | 示例代码展示如何使用 `RedisTemplate` 对 String 类型数据进行增删改查操作: ```java @SpringBootTest public class RedisTest { @Autowired private RedisTemplate<String, Object> redisTemplate; @Test public void testString() { // 写入一条 String 数据 redisTemplate.opsForValue().set("name", "wxp"); // 获取 string 数据 Object name = redisTemplate.opsForValue().get("name"); System.out.println("name = " + name); } } ``` --- #### 三、Redis 缓存在实际项目中的应用 为了提升系统的查询效率,在客户端与数据库之间引入 Redis 缓存是一种常见做法。具体流程如下: 1. **优先从 Redis 查询**:当接收到请求时,尝试从 Redis 中获取所需数据。 2. **未命中时回源查询**:如果 Redis 中不存在目标数据,则从 MySQL 或其他数据库中查询。 3. **更新缓存**:将查询到的结果同步写入 Redis,以便后续请求可以直接命中缓存[^2]。 示例代码展示了如何结合 Redis 实现商铺信息的快速查询功能: ```java @GetMapping("/{id}") public Result queryShopById(@PathVariable("id") Long id) { // 尝试从 Redis 中获取商铺信息 Shop shopFromCache = (Shop) redisTemplate.opsForValue().get("shop:" + id); if (shopFromCache != null) { return Result.ok(shopFromCache); // 如果命中缓存,直接返回结果 } // 如果 Redis 中无数据,则从数据库查询 Shop shopFromDb = shopService.getById(id); if (shopFromDb != null) { // 更新 Redis 缓存 redisTemplate.opsForValue().set("shop:" + id, shopFromDb); } return Result.ok(shopFromDb); } ``` --- #### 四、序列化与反序列化的处理 在 Spring Boot 应用中,默认情况下 Redis 使用的是 JdkSerializationRedisSerializer 来完成 keyvalue 的序列化/反序列化工作。然而,这种默认方式可能会导致 key 显示不直观的问题。因此,通常建议自定义 key 的序列化器,例如采用 `StringRedisSerializer`,以确保 key 能够正常显示[^3]。 示例代码演示如何设置 RedisTemplate 的序列化策略: ```java @Configuration public class RedisConfig { @Bean public RedisTemplate<String, Object> redisTemplate(RedisConnectionFactory factory) { RedisTemplate<String, Object> template = new RedisTemplate<>(); template.setKeySerializer(new StringRedisSerializer()); // 设置 Key 的序列化器 template.setValueSerializer(new GenericJackson2JsonRedisSerializer()); // 设置 Value 的序列化器 template.setConnectionFactory(factory); return template; } } ``` --- #### 五、总结 以上内容涵盖了 Redis 的基本概念、Spring Data Redis 的核心组件 `RedisTemplate` 的使用方法,以及 Redis 在实际开发中的典型应用场景——缓存优化。通过对这些知识点的学习,可以更好地掌握 Redis 技术并将其应用于生产环境。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值