使用spring-data-redis中的Redis事務(wù)
SessionCallback
Redis通過multi, exec, 或discard命令來提供事務(wù)支持,這些操作在RedisTemplate中同樣是可用的。
但是,RedisTemplate 默認(rèn)使用RedisCallBack接口,并不能保證使用同一連接來執(zhí)行同一事務(wù)中的所有操作(此時(shí)Transaction是無效的)。
又但是,Spring Data Redis提供了SessionCallback接口,以便在需要保證同一連接執(zhí)行多個(gè)操作時(shí)使用,比如“需要使用Redis事務(wù)時(shí)”。
我們能夠看到:
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行已經(jīng)做了連接綁定;
使用方式如下:
//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的所有結(jié)果。
另外一個(gè)額外的exec方法,允許您為事務(wù)結(jié)果傳遞自定義序列化器。
@Transactional支持
上面我們能夠看到,可以通過SessionCallback綁定連接,并且實(shí)現(xiàn)multi, exec,或discard,從而支持Redis事務(wù),但是這樣就顯得很復(fù)雜而且Redis操作(opsXXX.X)執(zhí)行的位置也變得有局限性(盡管不影響功能)。
然而,Spring下我們可以更加簡單,只需兩步:
- 為
method添加注解**@Transactional或者Xml配置**(< tx:method />),注冊事務(wù)切點(diǎn)。相當(dāng)于調(diào)用了TransactionSynchronizationManager.setActualTransactionActive(true); - 通過 setEnableTransactionSupport(true) 顯式啟用
RedisTemplate實(shí)例的事務(wù)支持(默認(rèn)被禁用)
/** 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實(shí)例 默認(rèn)調(diào)用 execute(RedisCallback action),方法內(nèi)容如下:
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開啟事務(wù)支持,同時(shí)transactionManager非只讀的實(shí)際事務(wù)被激活
if (enableTransactionSupport && isActualNonReadonlyTransactionActive()) {
connectionToBind = createConnectionProxy(conn, factory);
}
/**
* 不用管……
*/
return conn;
}可以看到,enableTransactionSupport = true 將會(huì)促使當(dāng)前Thread嘗試綁定RedisConnection,僅當(dāng)也 isActualNonReadonlyTransactionActive = true,連接才會(huì)成功綁定。
連接綁定成功,同時(shí)將會(huì)觸發(fā)MULTI。
一旦MULTI被調(diào)用:
- 當(dāng)前
RedisConnection將會(huì)排隊(duì)write操作; - 所有
readonly操作,例如KEYS將會(huì)被分發(fā)給一個(gè)全新的 (非Thread綁定)的RedisConnection; - 命令
EXEC或DISCARD將交由SpringAOP的動(dòng)態(tài)代理對象去調(diào)用: - 如果
事務(wù)構(gòu)建過程中沒有異常拋出(默認(rèn)RuntimeException及其子類),則EXEC被調(diào)用,執(zhí)行命令隊(duì)列; - 否則
DISCARD,清除命令隊(duì)列。
開啟事務(wù)支持后:
/** 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官網(wǎng)給出的,第三個(gè)顯然是WATCH命令開啟樂觀鎖后的結(jié)果。
然而至少在本人正在使用的 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命令并沒有被使用,親測第三種效果并不存在(你可以根據(jù)自己的依賴版本嘗試一下),此處亮出代碼。
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()未被調(diào)用
TransactionSynchronizationManager.registerSynchronization(new RedisTransactionSynchronizer(connHolder, conn,
factory));
}
}
}聲明兩個(gè)RedisTemplate實(shí)例
兩個(gè)RedisTemplate實(shí)例?
- 支持事務(wù):
commands要么統(tǒng)一執(zhí)行,要么都被清除,維護(hù)數(shù)據(jù)完整性; - 不支持事務(wù),
command立即執(zhí)行,即時(shí)返回執(zhí)行結(jié)果并且更高效;
/** 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;
}
}總結(jié)
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
- redis.clients.jedis.exceptions.JedisDataException:?NOAUTH?Authentication?required數(shù)據(jù)操作異常的解決方法
- redis.clients.jedis.exceptions.JedisAskDataException異常解決
- Spring Data Redis對象緩存序列化問題解決
- Spring?Data?Redis切換底層Jedis和Lettuce實(shí)現(xiàn)源碼解析
- 使用JMeter插件Redis Data Set如何實(shí)現(xiàn)高性能數(shù)據(jù)驅(qū)動(dòng)測試
相關(guān)文章
Spring?Boot面試必問之啟動(dòng)流程知識點(diǎn)詳解
SpringBoot是Spring開源組織下的子項(xiàng)目,是Spring組件一站式解決方案,主要是簡化了使用Spring的難度,簡省了繁重的配置,提供了各種啟動(dòng)器,開發(fā)者能快速上手,這篇文章主要給大家介紹了關(guān)于Spring?Boot面試必問之啟動(dòng)流程知識點(diǎn)的相關(guān)資料,需要的朋友可以參考下2022-06-06
SpringBoot如何訪問html和js等靜態(tài)資源配置
這篇文章主要介紹了SpringBoot如何訪問html和js等靜態(tài)資源配置,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-03-03
Eureka源碼閱讀之環(huán)境搭建及工程結(jié)構(gòu)
這篇文章主要為大家介紹了Eureka源碼閱讀之環(huán)境搭建的工程結(jié)構(gòu)及調(diào)試需知詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪<BR>2022-10-10
SpringBoot2 參數(shù)管理實(shí)踐之入?yún)⒊鰠⑴c校驗(yàn)的方式
這篇文章主要介紹了SpringBoot2 參數(shù)管理實(shí)踐,入?yún)⒊鰠⑴c校驗(yàn),本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-06-06
Java通過JNI調(diào)用C++動(dòng)態(tài)庫的完整流程詳解
JNI(Java Native Interface),是實(shí)現(xiàn)Java/Kotlin與C/C++語言之間交互的橋梁,本文主要為大家介紹了Java通過JNI調(diào)用C++動(dòng)態(tài)庫的完整流程,需要的可以參考下2025-04-04
spring boot切面execution表達(dá)式添加多個(gè)包路徑問題及解決方案
在Spring Boot中,如果你想為多個(gè)包中的方法創(chuàng)建一個(gè)切面,你可以在@Pointcut注解中使用||操作符來指定多個(gè)包,下面給大家分享spring boot切面execution表達(dá)式添加多個(gè)包路徑問題及解決方案,感興趣的朋友跟隨小編一起看看吧2024-03-03
Java代理的幾種實(shí)現(xiàn)方式總結(jié)
本文將通過例子說明java代理的幾種實(shí)現(xiàn)方式,并比較它們之間的差異,文中通過代碼示例給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作有一定的參考價(jià)值,需要的朋友可以參考下2023-12-12

