`
hello123654789
  • 浏览: 12656 次
文章分类
社区版块
存档分类
最新评论

Redis缓存技术学习系列之事务处理

阅读更多
在本系列的第一篇文章中,我们主要针对Redis中的“键”和“值”进行了学习。我们可以注意到,Redis是一个C/S架构的数据库,在我们目前的认知中,它是通过终端中的一条条命令来存储和读取的,即它是一个非常典型的“请求-响应”模型。可是我们知道在实际的应用中,我们要面对的或许是更为复杂的业务逻辑,因为Redis中不存在传统关系型数据库中表的概念,因此在使用Redis的过程中,我们要面对两个实际的问题,即如何更好的维护数据库中的”键“、如何在高效执行命令的同时保证命令执行成功。对于前者,我认为这是一个设计上的问题,而对于后者,我认为这是一个技术上的问题。所以,这篇文章的核心内容就是找到这两个问题的答案。带着这样的问题出发,我们就可以正式进入这篇文章的主题:Redis中的事务处理。

从数据库事务说起

​通常我们提及数据库都不可避免的要提到事务,那么什么是事务呢?事务是指作为单个逻辑工作单元执行的一系列操作。所以,首先事务是一系列操作,这一系列操作具有二态性,即完全地执行或者完全地不执行。因此事务处理可以确保除非事务单元内的所有操作的成功完成,否则不会永久更新面向数据的资源。我们这里举一个例子,数据库中除查询操作以外,插入(Insert)、删除(Delete)和更新(Update)这三种操作都会对数据造成影响,因为事务处理能够保证一系列操作可以完全地执行或者完全不执行,因此在一个事务被提交以后,该事务中的任何一条SQL语句在被执行的时候,都会生成一条撤销日志(Undo Log),而撤销日志中记录的是和当前擦作完全相反的操作,比如删除的相反操作是插入,插入的相反操作是删除等。我们通常所说的事务回滚其实就是去执行这些插销日志里的相反操作,这同样告诉我们一个道理,只有事务中的一系列操作完全执行的情况下可以回滚,如果是在意外情况下导致事务中的一系列操作没有完全执行,这个时候我们是不能保证数据一定可以回滚的。

​在数据库相关理论中,一个逻辑工作单元想要成为事务,就必须满足ACID,即原子性、一致性、隔离性和持久性。(1):原子性这个概念其实就是指,一个事务内的所有SQL操作都是一个整体,因此只有所有的SQL操作都完全执行成功,该事务方可以认为提交成功。如果在提交事务过程中某一条SQL语句执行失败,则整个事务必须回滚到事务提交前的状态。(2):而一致性这个概念则是指,事务在完成的时候,必须要保证所有的数据都保持一致的状态,而落实到数据库的各个组成部分上,则要求开发人员能够保证数据、索引、约束、日志等在事务前后具备一致性。(3):隔离性这个概念主要针对并发,其核心思想就是不同的并发事务对数据产生的修改必须是相互隔离的,假设有两个不同的事务A和B并发执行,那么对A来讲,它在执行前的状态只有两种,即B执行前和B执行后。同理,对B来讲同样是如此,这样的特性我们就称为隔离性。(4):持久性相对简单,是指事务完成以后它对数据的影响是永久性的。

Redis中的事务处理

​好了,截止到目前为止,我们对数据库中事务处理的相关理论有了一个基本的认识,或许这个世界上的数据库系统千差万别,但我相信在事务处理这个问题上它们最终会殊途同归,就像我们解决并发过程中的冲突问题,常规的做法依然是加锁一样,这是我之所以要花费精力去理解和解释这些理论知识的原因,技术可谓是日新月异,如果我们总是一味地为新技术而疲于奔命,那么或许我们会渐渐地失去对这个行业的热爱,我相信原理永远比框架更为重要,没有系统学习过计算机专业的课程,这件事情让我至今都颇为遗憾。Redis中的事务是可以视为一个队列,即我们可以通过MULTI开始一个事务,这相当于我们声明了一个命令队列。接下来,我们向Redis中提交的每条命令,都会被排入这个命令队列。当我们输入EXEC命令时,将触发当前事务,这相当于我们从命令队列中取出命令并执行,所以Redis中一个事务从开始到执行会经历开始事务、命令入队和执行事务三个阶段。下面是一个在Redis中使用事务的简单示例:

127.0.0.1:6379> MULTI
OK
127.0.0.1:6379> SET Book_Name "GIt Pro"
QUEUED
127.0.0.1:6379> SADD Program_Language "C++" "C#" "Jave" "Python"
QUEUED
127.0.0.1:6379> GET Book_Name
QUEUED
127.0.0.1:6379> EXEC
1) OK
2) (integer) 4
3) "GIt Pro"

我们可以注意到Redis中的事务和通常意义上的事务基本上是一致的,即

事务是由一系列操作组成的单个逻辑工作执行单元。特别地,因为在Redis中命令是存储在一个队列中,所以,事务中的所有命令都会按顺序执行,并且在执行事务的过程中不会被客户端发送的其它命令中断。
事务是一个原子操作,事物中的命令只有两种执行结果,即全部执行或者全部不执行。如果客户端在使用MULTI命令开启事务后因为意外而没有执行EXEC命令,则事务中的所有命令都不会执行。同理,如果客户端在使用MULTI命令开启事务后执行EXEC命令,则事务中的所有命令都会执行。
Redis中的事务可以使用DISCARD命令来清空一个命令队列,并放弃对事务的执行。如果命令在入队时发生错误,Redis将在客户端调用EXEC命令时拒绝执行并取消事务,但是在EXEC命令执行后发生的错误,Redis将选择自动忽略。
我们知道,常见的并发控制方案主要有悲观锁和乐观锁两种方案,这里首先来解释下这两种概念。所谓悲观锁,顾名思义是一种悲观的策略,悲观锁认为:在对任何记录做修改前都应该加锁,如果加锁失败则表明该机录正在被修改,此时应该抛出异常;如果加锁成功则修改记录并在事务完成后解锁;如果有其它人修改则应该等待当前修改解锁或者是抛出异常。而所谓乐观锁,顾名思义是一种乐观的策略,乐观锁认为:每次从记录中查找数据别人都不会修改,因此这个过程中不需要加锁,但是在更新记录的时候,会通过版本号来判断别人是否修改过当前记录。

通常来讲,乐观锁适合在写冲突相对较少的场合下,悲观锁适合在写冲突相对较多的场合下。Redis中提供了一种称为check-and-set的机制,该机制主要通过WATCH命令来实现,其原理正是基于乐观锁的策略,Redis会在执行EXEC命令前检查被监视的键对应的值是否发生变化,如果该值发生变化表明有人修改过这个键中存储的值,此时Redis将会自动取消当前事务。我们来看这个简单的例子:

WATCH Record_Count
val = GET Record_Count
val = val + 1
MULTI
SET Record_Count $val
EXEC

在这个例子中,我们尝试在事务中对Record_Count做一个自增操作,这段代码在非并发的情况下是没有问题的,可是在并发的情况下,如果在执行EXEC命令前有一个用户修改了Record_Count的值,那么我们此时的结果就会比期望的结果小1,现在我们有了WATCH,Redis就会对Record_Count进行监听,当Redis监听到该值发生变化的时候,这个事务就会被自动取消,进而避免造成冲突。

谈谈Redis中键的管理问题

​其实从切题的角度来讲,这篇博客基本上说清楚了事务处理问题,因此这篇博客虽然没有给大家带来多少惊喜,却依然可以非常恰到好处的结题,可是因为之前有朋友在博客中留言并问到Redis的键管理的问题,所以博主决定在这里简单的讨论下这个问题,鉴于博主和大家一样都是感刚接触Redis,所以下面的观点仅仅是一家之言,如果有疑问可以在博客中留言,欢迎大家批评指正。我认为Redis中的键的管理,基本上有两种策略,即惰性删除和定期删除,而实际上这正是Redis默认的键删除策略:

redis使用惰性删除和定期删除两种策略来删除过期的键:惰性删除策略在碰到过期键时方进行删除操作,定期删除策略则每隔一段时间主动查找并删除过期键。
所以,基于这两种键删除策略,我们可以想到的做法有:

对于临时变量可以采用临时键来存储,在数据库全局设定一个过期时间,由Redis在键过期后自动删除。
对于持久化数据可以采用普通键来存储,通过服务器和客户端间定义协议来由客户端主动删除键。
对于不同模块中的键采取统一规范的命名规则来命名键,从而解决Redis中键管理混乱的问题。
设计合理的键回收机制,避免Redis使用超过95%以上的内存,或者通过设置Redis中的最大内存容量及其内存策略来主动触发Redis对键的淘汰。具体可以参考:Sunnyxd - Redis学习笔记-事务、键空间的维护与性能
好了,这篇文章就是这样了,希望大家喜欢,下篇见!


源码来源: minglisoft.cn/technology
分享到:
评论

相关推荐

    redis本地缓存与redis缓存

    然而,由于涉及网络通信,Redis缓存的读写速度相对于本地缓存略慢,但可以通过设置合适的缓存策略(如LRU、LFU)和主从复制、分片等技术来优化性能。 在实际应用中,本地缓存通常用于短期、频繁访问且对数据一致性...

    redis缓存技术资料

    在缓存技术方面,Redis的优势体现在以下几个方面: 1. **高速读写**:由于Redis将数据存储在内存中,因此读写速度非常快,能够显著提高应用程序的响应速度,尤其对于频繁访问的热数据,Redis的缓存效果尤为明显。 ...

    使用.Net 技术操作Redis缓存技术,对Redis缓存进行存储增删改查等相关操作

    Redis是一种高性能的键值数据库,常用于数据缓存、消息队列、主从复制等多种场景。在.NET开发环境中,我们通常使用StackExchange.Redis库来与Redis进行交互。本篇文章将详细探讨如何使用.NET技术,尤其是C#语言,来...

    Redis缓存数据库技术

    Redis缓存数据库技术是现代Web开发中不可或缺的一部分,它的高效、灵活和丰富的功能使得它在数据缓存、消息传递、计数器、排行榜等多个领域得到广泛应用。通过学习和实践Redis,开发者可以提升应用的性能和用户体验...

    redis缓存技术学习.docx

    综上所述,Redis作为一个内存数据库,通过提供多种数据结构、持久化策略、主从复制和自定义的VM机制,成为了广泛应用的缓存技术。其高性能和丰富的数据操作使其在Web开发和大数据处理等领域中成为首选的工具之一。...

    MemCache和Redis缓存介绍

    在IT行业中,缓存技术是提高系统性能和响应速度的关键因素之一。MemCache和Redis是两种广泛使用的分布式内存缓存系统,它们能够有效地存储和检索数据,减轻数据库的负担,提升应用的性能。本文将深入探讨MemCache和...

    Redis缓存技术安装包文件.rar

    Redis,全称Remote Dictionary Server,是一款高性能的键值对存储系统,常被用作数据库、缓存和消息中间件。它的出现主要是为了解决在内存中存储数据的问题,以提供高速的数据读写性能。Redis支持丰富的数据结构,如...

    mybati与redis缓存demo合集

    标题"mybati与redis缓存demo合集"意味着我们将学习如何在MyBatis项目中集成Redis作为缓存机制,这通常涉及到以下几个步骤: 1. **安装与配置Redis**: 首先,你需要在服务器上安装Redis,并确保其正常运行。接着,在...

    基于redis的缓存框架

    对比Spring Cache,自定义的Redis缓存框架可能更专注于Redis特性的利用,例如使用Sorted Set进行范围查询,或者使用Hashes存储复杂对象,从而提高数据访问效率。此外,自定义框架可以根据项目需求进行更精细化的定制...

    redis-缓存文档

    ### Redis缓存技术详解 #### 一、Redis简介与缓存概念 Redis是一种高性能的开源键值存储系统,常被用作数据库、缓存和消息中间件。它支持多种数据结构,如字符串(strings)、散列(hashes)、列表(lists)、集合...

    基于mongodb数据库的集成redis缓存springboot实战

    在“基于mongodb数据库的集成redis缓存springboot实战”项目中,我们将学习如何将这两个技术结合在一起,以实现高效的数据存储和检索。首先,我们需要在SpringBoot项目中添加MongoDB和Redis的相关依赖。这通常通过在...

    ssm框架+shiro+redis缓存(某课网代码)

    本项目结合了Shiro安全框架和Redis缓存技术,进一步提升了应用的安全性和性能。 Shiro是Apache软件基金会下的一个开源项目,主要负责应用程序的安全管理,包括认证、授权、会话管理和密码加密等。它设计简洁,易于...

    springboot-redis事务

    在IT行业中,SpringBoot框架与Redis的集成是常见的应用场景,特别是在处理高并发、高性能的数据存储和缓存需求时。本文将深入探讨“SpringBoot-Redis事务”这一主题,旨在帮助开发者理解如何在SpringBoot项目中利用...

    SpringBoot项目+MybatisPlus使用+Redis缓存

    1. **MybatisPlus整合Redis**:在需要缓存的查询方法上添加`@Cacheable`注解,通过Spring Cache抽象层,将结果存储到Redis中。下次请求相同数据时,直接从缓存获取,提高响应速度。 2. **RedisTemplate与...

    spring + ehcache + redis两级缓存

    将Redis作为二级缓存,可以处理Ehcache无法容纳或者需要跨服务器共享的数据。 在Spring中集成Ehcache和Redis,我们需要做以下工作: 1. **配置Ehcache**: 首先,我们需要在项目中添加Ehcache的依赖,并创建一个...

    memcached完全剖析ehcache memcached redis 缓存技术总结

    标题中的“memcached完全剖析ehcache memcached redis 缓存技术总结”涵盖了三种常见的缓存技术:Memcached、Ehcache和Redis。这三种技术在IT行业中被广泛应用于提高应用程序性能,通过存储和检索频繁访问的数据来...

    spring+mybatis+redis缓存入门

    总的来说,"spring+mybatis+redis缓存入门"这个主题涵盖了Web应用开发中的重要技术栈。通过理解并实践这些步骤,你可以掌握如何利用这三大组件构建高性能、可扩展的应用,为你的职业生涯打下坚实的基础。记得在实际...

    Redis缓存中间件技术实践指南

    根据给定的信息,本文将详细解析“Redis缓存中间件技术实践指南”中涉及的关键知识点,包括Redis的特点、应用场景、支持的数据结构以及如何在不同平台上部署Redis。 ### 一、Redis概述 #### 1.1 Redis简介 Redis,...

    SSM整合Redis做缓存

    **实际应用场景**:SSM整合Redis缓存常见于高并发场景,如电商的商品信息、用户信息查询,减少数据库压力,提高响应速度。此外,还可以用于会话共享、消息队列等场景。 **注意事项**:在使用Redis做缓存时,需要...

    Java+redis缓存工具类(SSM)

    总之,这个项目提供了一个基于Java的Redis缓存解决方案,对于学习SSM框架整合Redis以及理解如何在实际项目中使用Redis作为缓存工具具有很好的参考价值。开发者可以通过研究这个项目,掌握如何在Java应用中高效地管理...

Global site tag (gtag.js) - Google Analytics