使用spring-data-redis中的Redis事务

 更新时间:2024年07月08日 09:40:01   作者:qq_32331073  
这篇文章主要介绍了使用spring-data-redis中的Redis事务,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

SessionCallback

Redis通过multi, exec, 或discard命令来提供事务支持,这些操作在RedisTemplate中同样是可用的。

但是,RedisTemplate 默认使用RedisCallBack接口,并不能保证使用同一连接来执行同一事务中的所有操作(此时Transaction是无效的)。

又但是,Spring Data Redis提供了SessionCallback接口,以便在需要保证同一连接执行多个操作时使用,比如“需要使用Redis事务时”。

我们能够看到:

public <T> T execute(SessionCallback<T> session) {
		Assert.isTrue(initialized, "template not initialized; call afterPropertiesSet() before using it");
		Assert.notNull(session, "Callback object must not be null");

		RedisConnectionFactory factory = getConnectionFactory();
		// bind connection
		RedisConnectionUtils.bindConnection(factory, enableTransactionSupport);//第8行
		try {
			return session.execute(this);
		} finally {
			RedisConnectionUtils.unbindConnection(factory);
		}
	}

RedisTemplate.execute(SessionCallback<T> session)方法的第8行已经做了连接绑定

使用方式如下:

//execute a transaction
List<Object> txResults = redisTemplate.execute(new SessionCallback<List<Object>>() {
  public List<Object> execute(RedisOperations operations) throws DataAccessException {
    operations.multi();
    operations.opsForSet().add("key", "value1");

    // This will contain the results of all ops in the transaction
    return operations.exec();
  }
});
System.out.println("Number of items added to set: " + txResults.get(0));

在返回之前,RedisTemplate将使用它的value, hash key和hash value 序列化器来反序列化exec的所有结果。

另外一个额外的exec方法,允许您为事务结果传递自定义序列化器。

@Transactional支持

上面我们能够看到,可以通过SessionCallback绑定连接,并且实现multi, exec,或discard,从而支持Redis事务,但是这样就显得很复杂而且Redis操作(opsXXX.X)执行的位置也变得有局限性(尽管不影响功能)。

然而,Spring下我们可以更加简单,只需两步:

  • method 添加注解**@Transactional或者Xml配置**(< tx:method />),注册事务切点。相当于调用了TransactionSynchronizationManager.setActualTransactionActive(true);
  • 通过 setEnableTransactionSupport(true) 显式启用RedisTemplate实例的事务支持默认被禁用
/** Sample Configuration **/
@Configuration
public class RedisTxContextConfiguration {
  @Bean
  public StringRedisTemplate redisTemplate() {
    StringRedisTemplate template = new StringRedisTemplate(redisConnectionFactory());
    // explicitly enable transaction support
    template.setEnableTransactionSupport(true);
    return template;
  }
}

redisTemplate实例 默认调用 execute(RedisCallback action),方法内容如下:

public <T> T execute(RedisCallback<T> action, boolean exposeConnection, boolean pipeline){
		/**
		 * 变量声明等操作……
		 */
		try {
			if (enableTransactionSupport) {
				// only bind resources in case of potential transaction synchronization
				conn = RedisConnectionUtils.bindConnection(factory, enableTransactionSupport);
			} else {
				conn = RedisConnectionUtils.getConnection(factory);
			}
		/**
		 * 其他操作……
		 */
}

public static RedisConnection bindConnection(RedisConnectionFactory factory, 
			boolean enableTransactionSupport) {
		/**
		 * 不用管……
		 */
		RedisConnection conn = factory.getConnection();
		RedisConnection connectionToBind = conn;
		//redisTemplate开启事务支持,同时transactionManager非只读的实际事务被激活
		if (enableTransactionSupport && isActualNonReadonlyTransactionActive()) {
			connectionToBind = createConnectionProxy(conn, factory);
		}
		/**
		 * 不用管……
		 */
		return conn;
}

可以看到,enableTransactionSupport = true 将会促使当前Thread尝试绑定RedisConnection,仅当也 isActualNonReadonlyTransactionActive = true,连接才会成功绑定。

连接绑定成功,同时将会触发MULTI

一旦MULTI被调用:

  • 当前RedisConnection将会排队write操作
  • 所有readonly操作,例如KEYS将会被分发给一个全新的 (非Thread绑定)的RedisConnection
  • 命令EXECDISCARD将交由SpringAOP动态代理对象去调用:
  • 如果事务构建过程中没有异常抛出(默认RuntimeException及其子类),则EXEC被调用,执行命令队列;
  • 否则DISCARD,清除命令队列。

开启事务支持后:

/** Usage Constrainsts **/
// executed on thread bound connection
template.opsForValue().set("foo", "bar");

// read operation executed on a free (not tx-aware)
connection template.keys("*");

// returns null as values set within transaction are not visible
template.opsForValue().get("foo");

上面的样例代码是Spring官网给出的,第三个显然是WATCH命令开启乐观锁后的结果。

然而至少在本人正在使用的 spring-data-redis-1.8.10.RELEASE.jar

<dependency>
	<groupId>org.springframework.data</groupId>
	<artifactId>spring-data-redis</artifactId>
	<version>1.8.10.RELEASE</version>
</dependency>

WATCH命令并没有被使用,亲测第三种效果并不存在(你可以根据自己的依赖版本尝试一下),此处亮出代码。

org.springframework.data.redis.core.RedisConnectionUtils.potentiallyRegisterTransactionSynchronisation

private static void potentiallyRegisterTransactionSynchronisation(RedisConnectionHolder connHolder,
			final RedisConnectionFactory factory) {

		if (isActualNonReadonlyTransactionActive()) {

			if (!connHolder.isTransactionSyncronisationActive()) {
				connHolder.setTransactionSyncronisationActive(true);

				RedisConnection conn = connHolder.getConnection();
				conn.multi();//在此之前conn.watch()未被调用

				TransactionSynchronizationManager.registerSynchronization(new RedisTransactionSynchronizer(connHolder, conn,
						factory));
			}
		}
	}

声明两个RedisTemplate实例

两个RedisTemplate实例?

  • 支持事务:commands要么统一执行,要么都被清除,维护数据完整性;
  • 不支持事务,command立即执行,即时返回执行结果并且更高效;
/** Sample Configuration **/
@Configuration
public class RedisTxContextConfiguration {
  @Bean
  public StringRedisTemplate redisTransactionTemplate() {
    StringRedisTemplate template = new StringRedisTemplate(redisConnectionFactory());
    // explicitly enable transaction support
    template.setEnableTransactionSupport(true);
    return template;
  }
 @Bean
  public StringRedisTemplate redisTemplate() {
    StringRedisTemplate template = new StringRedisTemplate(redisConnectionFactory());
    return template;
  }
}

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • java 线程创建多线程详解

    java 线程创建多线程详解

    本文主要讲解java 线程创建多线程的知识,这里对java线程的创建做了详细介绍,并附简单示例代码,有兴趣的小伙伴可以参考下
    2016-09-09
  • Java的Lambda表达式使用及说明

    Java的Lambda表达式使用及说明

    文章介绍了JavaSE8中的Lambda表达式,其基本语法、函数接口的概念以及在集合中的使用,同时,还讨论了Lambda表达式的变量捕获和一些优点与缺点
    2026-02-02
  • 彻底搞懂Java多线程(四)

    彻底搞懂Java多线程(四)

    这篇文章主要给大家介绍了关于Java面试题之多线程和高并发的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用java具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2021-07-07
  • 基于Java ORM框架的使用详解

    基于Java ORM框架的使用详解

    本篇文章是对Java中ORM框架的使用进行了详细的分析介绍,需要的朋友参考下
    2013-05-05
  • Java中jqGrid 学习笔记整理——进阶篇(二)

    Java中jqGrid 学习笔记整理——进阶篇(二)

    这篇文章主要介绍了Java中jqGrid 学习笔记整理——进阶篇(二)的相关资料,需要的朋友可以参考下
    2016-04-04
  • 详解如何使用IntelliJ IDEA新建一个Servlet项目

    详解如何使用IntelliJ IDEA新建一个Servlet项目

    这篇文章主要介绍了详解如何使用IntelliJ IDEA新建一个Servlet项目,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-11-11
  • SpringBoot整合通用mapper实现泛型BaseController增删改方式

    SpringBoot整合通用mapper实现泛型BaseController增删改方式

    本文介绍了如何在SpringBoot项目中整合通用Mapper实现泛型BaseController的增删改功能,通过统一的代码实现减少重复,提高开发效率
    2025-11-11
  • 多模块项目引入SpringSecurity后一直报404的解决方案

    多模块项目引入SpringSecurity后一直报404的解决方案

    这篇文章主要介绍了多模块项目引入SpringSecurity后一直报404的解决方案,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-06-06
  • Java设计模式之装饰模式原理与用法实例详解

    Java设计模式之装饰模式原理与用法实例详解

    这篇文章主要介绍了Java设计模式之装饰模式原理与用法,结合实例形式详细分析了装饰模式的概念、原理、定义与使用方法,并总结分析了装饰模式的优缺点,具有一定参考借鉴价值,需要的朋友可以参考下
    2018-04-04
  • Java调用微信支付功能的方法示例代码

    Java调用微信支付功能的方法示例代码

    这篇文章主要介绍了Java调用微信支付功能的方法示例代码,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-08-08

最新评论