2024-09-05

Redis是一个开源的使用C语言编写的、支持网络交互的、可基于内存也可持久化的日志型、Key-Value数据库,并提供多种语言的API。

Redis基本数据类型包括:String、Hash、List、Set、Sorted Set、Bitmap、HyperLogLog、Geo、Stream。

以下是各种数据类型的基本操作命令:

  1. String类型



# 设置key-value
SET key value
 
# 获取key的value
GET key
 
# 删除key
DEL key
 
# 设置key的过期时间(单位:秒)
EXPIRE key seconds
  1. Hash类型



# 设置hash field为指定值,如果key不存在,则先创建
HSET key field value
 
# 获取key的hash field的value
HGET key field
 
# 删除key的hash field
HDEL key field
  1. List类型



# 在list头部插入值
LPUSH key value
 
# 在list尾部插入值
RPUSH key value
 
# 获取list中start到end的元素(-1表示最后一个元素)
LRANGE key start end
 
# 删除list中count个value
LREM key count value
  1. Set类型



# 添加一个或多个成员到set中
SADD key member
 
# 获取set中所有成员
SMEMBERS key
 
# 删除set中的一个或多个成员
SREM key member
  1. Sorted Set类型



# 添加一个或多个成员,或更新已存在成员的分数
ZADD key score member
 
# 获取sorted set中start到end的成员(-1表示最后一个成员)
ZRANGE key start end
 
# 删除sorted set中的一个或多个成员
ZREM key member
  1. Bitmap



# 对key中offset位置的bit进行设置
SETBIT key offset value
 
# 获取key中offset位置的bit值
GETBIT key offset
  1. HyperLogLog



# 添加一个或多个元素到HyperLogLog中
PFADD key element
 
# 获取HyperLogLog的基数
PFCOUNT key
  1. Geo



# 添加地理位置信息
GEOADD key longitude latitude member
 
# 获取两个地理位置的距离
GEODIST key member1 member2
 
# 获取指定位置的周边位置
GEORADIUS key longitude latitude radius m|km|ft|mi
  1. Stream



# 添加一条或多条记录到stream
XADD key ID field value
 
# 获取stream中的记录
XRANGE key start end
 
# 删除stream中的记录
XDEL key ID

以上操作是Redis基础操作,在实际应用中,还可以使用Redis的事务、发布/订阅模式、Lua脚本等高级特性。

2024-09-05

Redis集群是一种Redis服务器之间的网络连接,其中数据自动在多个节点之间分区。以下是一些可以用来设置和操作Redis集群的基本命令和示例代码。

  1. 创建Redis集群

Redis集群的创建通常需要多个Redis实例。你可以在每个实例上运行相同的redis-server命令,并为它们提供不同的端口号。例如,你可以使用以下命令在端口7000、7001和7002上创建三个Redis实例:




redis-server --port 7000 --cluster-enabled yes --cluster-config-file nodes-7000.conf --cluster-node-timeout 5000 --appendonly yes --appendfilename appendonly-7000.aof --dbfilename dump-7000.rdb
redis-server --port 7001 --cluster-enabled yes --cluster-config-file nodes-7001.conf --cluster-node-timeout 5000 --appendonly yes --appendfilename appendonly-7001.aof --dbfilename dump-7001.rdb
redis-server --port 7002 --cluster-enabled yes --cluster-config-file nodes-7002.conf --cluster-node-timeout 5000 --appendonly yes --appendfilename appendonly-7002.aof --dbfilename dump-7002.rdb
  1. 创建集群

一旦你有了多个Redis实例在运行,你可以使用redis-cli工具创建集群。以下命令将创建一个包含在端口7000、7001和7002上运行的Redis实例的集群:




redis-cli --cluster create 127.0.0.1:7000 127.0.0.1:7001 127.0.0.1:7002 --cluster-replicas 1
  1. 查看集群信息

要查看集群的信息,你可以使用以下命令:




redis-cli -c -p 7000 cluster info
  1. 添加节点

如果你想向现有集群添加一个新节点,你可以首先启动一个新的Redis实例,然后使用以下命令将其添加到集群:




redis-cli --cluster add-node 127.0.0.1:7003 127.0.0.1:7000
  1. 分配槽

在Redis集群中,数据是根据哈希槽分区的。你可以使用以下命令手动分配或重新分配哈希槽:




redis-cli --cluster reshard 127.0.0.1:7000
  1. 故障转移

如果主节点失败,Redis集群将自动启动故障转移进程。你也可以手动启动故障转移:




redis-cli --cluster failover 127.0.0.1:7000
  1. 删除节点

如果你需要从集群中删除一个节点,你可以首先将该节点上的所有键迁移到其他节点,然后删除它:




redis-cli --cluster del-node 127.0.0.1:7000 <node-id-of-node-to-be-deleted>

以上命令和操作是Redis集群管理的基础。在实际应用中,你可能需要更复杂的脚本和工具来管理你的Redis集群。

2024-09-05

在Java中,使用Jedis客户端可以通过实现MessageListener接口来订阅Redis的频道消息。以下是一个简单的例子:




import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPubSub;
 
public class RedisMessageSubscriber {
 
    public static void main(String[] args) {
        Jedis jedis = new Jedis("localhost");
        JedisPubSub listener = new Listener();
 
        // 订阅一个频道
        jedis.subscribe(listener, "channel1");
    }
 
    static class Listener extends JedisPubSub {
 
        @Override
        public void onMessage(String channel, String message) {
            System.out.println("Received Message: " + message + " on channel: " + channel);
        }
 
        @Override
        public void onSubscribe(String channel, int subscribedChannels) {
            System.out.println("Subscribed to channel: " + channel + ", Total subscribed channels: " + subscribedChannels);
        }
 
        @Override
        public void onUnsubscribe(String channel, int subscribedChannels) {
            System.out.println("Unsubscribed from channel: " + channel + ", Total subscribed channels: " + subscribedChannels);
        }
    }
}

在这个例子中,我们创建了一个Jedis实例来连接Redis服务器,然后创建了一个继承自JedisPubSubListener类,并覆盖了onMessageonSubscribeonUnsubscribe方法来处理不同的订阅事件。最后,我们使用subscribe方法来订阅名为"channel1"的频道。

当有消息发布到"channel1"时,Listener类中的onMessage方法会被调用,并打印出接收到的消息和频道。同样,当订阅或取消订阅频道时,onSubscribeonUnsubscribe方法会被调用,并打印出相关信息。

2024-09-05

报错解释:

NOAUTH 错误表示客户端尝试执行命令,但是没有通过认证。在使用 Spring Boot 配置 Redis 时,如果 Redis 设置了密码,且 Spring Boot 配置中没有正确指定这个密码,就会发生这个错误。

解决方法:

  1. 检查你的 Redis 服务器是否设置了密码。如果设置了,你需要在 Spring Boot 的配置文件中指定这个密码。
  2. 如果你使用的是 application.propertiesapplication.yml 文件配置 Redis,确保添加了正确的密码配置。

对于 application.properties 文件,添加如下配置:




spring.redis.password=你的密码

对于 application.yml 文件,添加如下配置:




spring:
  redis:
    password: 你的密码
  1. 如果你使用的是 Lettuce 作为客户端连接池,确保在配置中正确设置了 spring.redis.lettuce.password 属性。
  2. 确保没有配置错误,比如拼写错误或多余的空格等。
  3. 如果你使用的是集群模式,确保所有节点的密码都设置一致。
  4. 重启 Spring Boot 应用,以应用新的配置。

如果以上步骤正确完成,应该能解决 NOAUTH 错误。

2024-09-05



# 假设有一个Redis数据库,包含以下键和值:
# 键: "user:1",值: "John Doe"
# 键: "user:2",值: "Jane Smith"
# 键: "age:1",值: "25"
# 键: "age:2",值: "30"
 
# 1. 使用Redis命令获取用户名
# 假设我们需要获取用户ID为1的用户名
GET user:1
 
# 2. 使用Redis命令获取用户年龄
# 假设我们需要获取用户ID为1的年龄
GET age:1
 
# 3. 使用Redis事务来同时设置用户名和年龄
# 假设我们需要同时为用户ID为1设置新的用户名和年龄
MULTI
SET user:1 "Jim Beam"
SET age:1 "28"
EXEC
 
# 4. 使用Redis的列表操作存储用户的购物车信息
# 假设用户ID为1的购物车包含商品ID为1001和1002
RPUSH cart:1 1001 1002
 
# 5. 使用Redis的集合操作跟踪关注者和被关注者
# 假设用户ID为1的关注者为用户ID 2和3,被关注者为用户ID 4和5
SADD followers:1 2 3
SADD following:1 4 5
 
# 6. 使用Redis的有序集合操作存储用户的排行榜分数
# 假设用户ID为1的排行榜分数为1000
ZADD leaderboard 1000 1
 
# 7. 使用Redis的发布/订阅模式实现消息推送
# 假设我们需要向所有订阅者发送一条新闻
PUBLISH news "New release available"
 
# 8. 使用Redis的Lua脚本来原子化更新用户信息
# 假设我们需要同时更新用户ID为1的用户名和年龄
EVAL "local userId = KEYS[1] return redis.call('SET', 'user:1', ARGV[1]) and redis.call('SET', 'age:1', ARGV[2])" 1 user:1 "Jim Beam" "28"
 
# 9. 使用Redis的GEO数据类型存储地理位置信息
# 假设有一个用户ID为1的用户,其地理位置为经度12.34和纬度56.78
GEOADD user_locations 12.34 56.78 1
 
# 10. 使用Redis的过期特性设置键的有效期
# 假设我们需要设置用户ID为1的用户名在10分钟后过期
SETEX user:1 600 "Jim Beam"
 
# 11. 使用Redis的SCAN命令迭代所有的键
# 假设我们需要迭代所有的用户键
SCAN 0 MATCH user:*
 
# 12. 使用Redis的BIT操作来跟踪用户的访问情况
# 假设用户ID为1在特定日期有访问记录
SETBIT user_visits:1 20230301 1
 
# 13. 使用Redis的HyperLogLog来估算页面访问量
# 假设用户ID为1访问了一个页面
PFADD page_visits:homepage <user id>
 
# 14. 使用Redis的布隆过滤器来检查元素是否可能存在
# 假设我们需要检查用户ID为1是否可能为
2024-09-05

Webdis 是一个使用 C 语言编写的项目,它提供了一个 HTTP 接口到 Redis 的一个子集命令集。这样的接口对于不能直接使用 Redis 协议的应用程序非常有用,例如 web 应用程序。

以下是一个简单的 Python 示例,使用 requests 库来调用 Webdis 服务:




import requests
 
# 设置 Webdis 服务器的 URL
url = 'http://localhost:7379'
 
# 发送 GET 请求来获取 key 的值
response = requests.get(f'{url}/get/mykey')
 
# 打印返回的内容
print(response.text)
 
# 发送 SET 请求来设置 key 的值
requests.get(f'{url}/set/mykey/myvalue')
 
# 再次获取 key 的值以确认设置成功
response = requests.get(f'{url}/get/mykey')
print(response.text)

在这个例子中,我们首先尝试获取键 mykey 的值,然后我们设置它的值为 myvalue,并再次尝试获取以确认它确实被设置了。

请注意,Webdis 并不支持所有 Redis 命令,而且它不是为了高性能设计的,因此它可能不适合需要高吞吐量的生产环境。对于这些情况,你可能需要考虑使用 Redis 的原生协议支持,或者使用代理服务器来转发请求。

2024-09-05



import redis
 
# 假设已经有一个Redis实例运行在本地,默认端口6379
redis_host = "localhost"
redis_port = 6379
 
# 创建一个Redis连接
redis_client = redis.StrictRedis(host=redis_host, port=redis_port, decode_responses=True)
 
# 设置热点数据的键值对
def set_hotspot_data(key, value, ttl=3600):
    # 使用SET命令设置键值对,并设置过期时间
    redis_client.setex(key, ttl, value)
 
# 获取热点数据
def get_hotspot_data(key):
    # 使用GET命令获取键对应的值
    return redis_client.get(key)
 
# 示例:设置并获取一个热点数据
set_hotspot_data("hot_news", "Bitcoin price soars to new highs")
hot_news_data = get_hotspot_data("hot_news")
print(hot_news_data)  # 输出: Bitcoin price soars to new highs

这段代码展示了如何使用Python和Redis库来设置和获取热点数据。在这个例子中,我们使用了setex命令来一次性设置键值对和过期时间,并使用get命令来获取键对应的值。这样的操作对于缓存热点数据非常高效,并且能够显著减少数据库的访问压力。

2024-09-05



import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.data.redis.connection.ReactiveRedisConnectionFactory;
import org.springframework.data.redis.core.ReactiveRedisTemplate;
import org.springframework.data.redis.serializer.GenericJackson2JsonRedisSerializer;
import org.springframework.data.redis.serializer.RedisSerializationContext;
 
@Configuration
public class ReactiveRedisConfig {
 
    @Bean
    public ReactiveRedisTemplate<String, Object> reactiveRedisTemplate(ReactiveRedisConnectionFactory factory) {
        return new ReactiveRedisTemplate<>(factory, RedisSerializationContext.java().string().serializer(new GenericJackson2JsonRedisSerializer()));
    }
}

这段代码定义了一个配置类ReactiveRedisConfig,其中包含一个Bean方法reactiveRedisTemplate,用于创建一个ReactiveRedisTemplate实例,该实例使用字符串作为键的序列化方式,并且使用GenericJackson2JsonRedisSerializer作为值的序列化器。这样配置后,你可以直接在Spring Data Reactive Redis中使用这个模板来操作Redis,无需手动序列化和反序列化对象。

2024-09-05

解释:

  1. 缓存穿透:查询不存在的数据,缓存不命中,请求直接打到数据库,可能导致数据库压力剧增。
  2. 缓存雪崩:缓存失效时间集中,导致大量请求打到数据库。
  3. 淘汰策略:当内存不足时,根据一定策略移除缓存数据。

解决方案:

  1. 缓存穿透:使用布隆过滤器,维护一个全局集合,存储所有可能的查询key。访问前先经过布隆过滤器,如果不存在则直接拒绝请求。
  2. 缓存雪崩:

    • 设置缓存数据的过期时间随机化,避免大量数据同时失效。
    • 实现热点数据永不过期或使用二级缓存。
  3. 淘汰策略:

    • 使用LRU(最近最少使用)、LFU(最常使用)、FIFO(先进先出)等策略。
    • 当内存不足时,通过设置内存淘汰机制来保证业务的正常运行。

示例代码(假设使用Redis作为缓存):




# 缓存key的存在性检查
def is_key_exist(key):
    # 假设有一个布隆过滤器实例
    return bloom_filter.is_exist(key)
 
# 设置缓存,并设置随机过期时间
def set_cache(key, value, ttl):
    random_ttl = random.randint(ttl//2, ttl)
    redis.setex(key, random_ttl, value)
 
# 淘汰策略
redis.config_set("maxmemory-policy", "allkeys-lru") # 设置LRU淘汰策略

注意:具体实施时需要根据业务场景和需求进行调整。

2024-09-04

在使用Sa-Token进行短信登录、注册以及鉴权时,可以结合Redis来存储短信验证码以及用户会话信息,以下是一个简化的示例代码:

首先,需要添加Sa-Token和Redis的依赖:




<!-- Sa-Token 依赖 -->
<dependency>
    <groupId>cn.dev33</groupId>
    <artifactId>sa-token-spring-boot-starter</artifactId>
    <version>最新版本</version>
</dependency>
<!-- Redis 依赖 -->
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-redis</artifactId>
</dependency>

然后配置Sa-Token与Redis的集成:




@Configuration
public class SaTokenConfig {
    @Bean
    public SaTokenAction saTokenAction() {
        return new SaTokenActionRedis(); // 使用Redis作为存储方式
    }
}

接下来,实现短信登录和注册的接口:




@RestController
public class MessageController {
 
    // 发送短信验证码接口
    @PostMapping("/send-message")
    public Response sendMessage(@RequestParam("phone") String phone) {
        // 生成验证码并保存至Redis
        String code = RandomUtil.randomNumbers(6);
        SaTokenRedisUtil.set(SaTokenRedisKeyBuild.smsCodeKey(phone), code, 1000 * 60 * 5); // 有效期5分钟
 
        // 模拟发送短信操作
        // sendSms(phone, code);
        return Response.success("短信发送成功");
    }
 
    // 短信登录接口
    @PostMapping("/sms-login")
    public Response smsLogin(@RequestParam("phone") String phone, @RequestParam("code") String code) {
        // 从Redis获取验证码并校验
        String cacheCode = SaTokenRedisUtil.get(SaTokenRedisKeyBuild.smsCodeKey(phone));
        if (!code.equals(cacheCode)) {
            return Response.error("验证码错误");
        }
 
        // 登录成功,创建会话
        StpUtil.login(phone); // 更多StpUtil方法参考Sa-Token官方文档
        return Response.success("登录成功");
    }
 
    // 短信注册接口
    @PostMapping("/sms-register")
    public Response smsRegister(@RequestParam("phone") String phone, @RequestParam("code") String code) {
        // 验证码校验同登录接口
        // ...
 
        // 注册用户
        int userId = StpUtil.getLoginIdAsInt(); // 假设用户ID是自增的,实际应用中需要根据业务逻辑生成
        // 保存用户信息至数据库等操作
        // saveUser(userId, phone);
 
        return Response.success("注册成功");
    }
}

以上代码提供了短信验证码的发送、登录和注册接口,并使用Redis来存储验证码和会话信息。在实际应用中,你需要根据自己的业务逻辑来扩展这些接口,比如集成短信服务提供商的API,处理用户信息的存储等。