- 浏览: 1198 次
-
文章分类
最新评论
在spring应用层扩展DataSourceTransactionManager ,判读事务属性readOnly实现读写分离
import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.jdbc.datasource.DataSourceTransactionManager; import org.springframework.transaction.TransactionDefinition; import org.springframework.transaction.support.DefaultTransactionStatus; /** * 1.事务管理由 TransactionInterceptor 拦截,执行invoke * 2.调用 TransactionAspectSupport#invokeWithinTransaction 实现环绕通知 * 3.getTransaction>>doGetTransaction>>isExistingTransaction{ * return handleExistingTransaction * } * 4.if>>[PROPAGATION_REQUIRED || PROPAGATION_REQUIRES_NEW || PROPAGATION_NESTED]{ * doBegin 在此将创建Connection,如果Connection 为新创建,绑定到TransactionSynchronizationManager#bindResource(DataSource, ConnectionHolder); * prepareTransactionStatus * return * } * 5.prepareTransactionStatus 将当前事务事务绑定到本地线程 * 如果第四步没有执行,Connection 将由 DataSourceUtils#doGetConnection 创建,绑定到TransactionSynchronizationManager#bindResource(DataSource, ConnectionHolder) * 6.执行ReflectiveMethodInvocation#proceed 如果抛出异常执行completeTransactionAfterThrowing处理 * 7.cleanupTransactionInfo * 8.commitTransactionAfterReturning>TransactionManager#commit * * Created by ShiHui on 2016/1/9. */ public class DataSourceProxyTransactionManager extends DataSourceTransactionManager { private static Logger logger = LoggerFactory.getLogger(DataSourceProxyTransactionManager.class); @Override protected Object doGetTransaction() { logger.debug(">>> doGetTransaction."); return super.doGetTransaction(); } /** * @param transaction * @param definition PROPAGATION_REQUIRED || PROPAGATION_REQUIRES_NEW || PROPAGATION_NESTED */ @Override protected void doBegin(Object transaction, TransactionDefinition definition) { logger.info(">>> doBegin transaction : {}"); logger.info(">>> TransactionDefinition[name = {}, level = {}, propagation = {}, isReadOnly = {}]" ,definition.getName() ,getIsolationLevelName(definition.getIsolationLevel()) ,getPropagationBehaviorName(definition.getPropagationBehavior()) ,definition.isReadOnly()); determineDataSource(definition); super.doBegin(transaction, definition); } /** * Initialize transaction synchronization as appropriate. */ @Override protected void prepareSynchronization(DefaultTransactionStatus status, TransactionDefinition definition) { logger.info(">>> prepareSynchronization status : {}, definition : {}", status, definition); determineDataSource(definition); super.prepareSynchronization(status, definition); } private void determineDataSource(TransactionDefinition definition){ if(DataSourceProxyManager.isNone()){ if(definition.isReadOnly()){ DataSourceProxyManager.markSlave(); logger.debug(">>> markSlave because readOnly = {}", definition.isReadOnly()); }else{ DataSourceProxyManager.markMaster(); logger.debug(">>> markMaster because readOnly = {}", definition.isReadOnly()); } } } @Override protected void doCommit(DefaultTransactionStatus status) { logger.debug(">>> doCommit"); super.doCommit(status); } @Override protected void doRollback(DefaultTransactionStatus status) { logger.debug(">>> doRollback"); super.doRollback(status); } @Override protected void doResume(Object transaction, Object suspendedResources) { logger.debug(">>> doResume"); super.doResume(transaction, suspendedResources); } @Override protected Object doSuspend(Object transaction) { logger.debug(">>> doSuspend"); return super.doSuspend(transaction); } @Override protected void doCleanupAfterCompletion(Object transaction) { DataSourceProxyManager.rest(); logger.debug(">>> doCleanupAfterCompletion before rest DataSource STATUS"); super.doCleanupAfterCompletion(transaction); } @Override protected boolean isExistingTransaction(Object transaction) { boolean isExisting = super.isExistingTransaction(transaction); logger.debug(">>> isExistingTransaction : {}", isExisting); return isExisting; } private String getIsolationLevelName(int code){ switch (code){ case TransactionDefinition.ISOLATION_DEFAULT: return "ISOLATION_DEFAULT"; case TransactionDefinition.ISOLATION_READ_COMMITTED: return "ISOLATION_READ_COMMITTED"; case TransactionDefinition.ISOLATION_READ_UNCOMMITTED: return "ISOLATION_READ_UNCOMMITTED"; case TransactionDefinition.ISOLATION_REPEATABLE_READ: return "ISOLATION_REPEATABLE_READ"; case TransactionDefinition.ISOLATION_SERIALIZABLE: return "ISOLATION_SERIALIZABLE"; default: return "ISOLATION_DEFAULT"; } } private String getPropagationBehaviorName(int code){ switch (code){ case TransactionDefinition.PROPAGATION_MANDATORY: return "PROPAGATION_MANDATORY"; case TransactionDefinition.PROPAGATION_NESTED: return "PROPAGATION_NESTED"; case TransactionDefinition.PROPAGATION_NEVER: return "PROPAGATION_NEVER"; case TransactionDefinition.PROPAGATION_NOT_SUPPORTED: return "PROPAGATION_NOT_SUPPORTED"; case TransactionDefinition.PROPAGATION_REQUIRED: return "PROPAGATION_REQUIRED"; case TransactionDefinition.PROPAGATION_REQUIRES_NEW: return "PROPAGATION_REQUIRES_NEW"; case TransactionDefinition.PROPAGATION_SUPPORTS: return "PROPAGATION_SUPPORTS"; default: return "PROPAGATION_SUPPORTS"; } } } //#################################### /** * * @author Stony * */ public class DataSourceProxyManager { private enum DataSourceType{ MASTER,SLAVE,ALWAYS_MASTER } private static ThreadLocal<DataSourceType> holder = new ThreadLocal<DataSourceType>(); public static boolean isMaster(){ return DataSourceType.ALWAYS_MASTER == holder.get() || DataSourceType.MASTER == holder.get(); } public static boolean isNone(){ return null == holder.get(); } public static void markSlave() { holder.set(DataSourceType.SLAVE); } public static void markMaster(){ holder.set(DataSourceType.MASTER); } public static void rest(){ holder.set(null); } public static void alwaysMaster(){ holder.set(DataSourceType.ALWAYS_MASTER); } } ///############################# import org.slf4j.Logger; import org.slf4j.LoggerFactory; import org.springframework.beans.factory.InitializingBean; import org.springframework.jdbc.datasource.AbstractDataSource; import org.springframework.util.Assert; import javax.sql.DataSource; import java.sql.Connection; import java.sql.SQLException; import java.util.ArrayList; import java.util.Collections; import java.util.List; import java.util.concurrent.atomic.AtomicInteger; /** * Created by ShiHui on 2016/1/9. */ public class DataSourceProxy extends AbstractDataSource implements InitializingBean{ private static Logger logger = LoggerFactory.getLogger(DataSourceProxy.class); private DataSource master; private List<DataSource> slaves; private int slavesCount = 0; private AtomicInteger slaveRequest = new AtomicInteger(1); public DataSource getMaster() { return master; } public void setMaster(DataSource master) { this.master = master; } public List<DataSource> getSlaves() { return slaves; } public void setSlave(DataSource slave) { if(getSlaves() == null){ setSlaves(Collections.synchronizedList(new ArrayList<DataSource>())); } getSlaves().add(slave); } public void setSlaves(List<DataSource> slaves) { this.slaves = slaves; } private DataSource determineDataSource(){ if(DataSourceProxyManager.isNone()){ logger.debug(">>> STATUS isNone current determine db is master"); return this.master; } if(DataSourceProxyManager.isMaster()){ logger.debug(">>> STATUS isMaster current determine db is master"); return this.master; } return determineSlaveDataSource(); } private DataSource determineSlaveDataSource() { int index = slaveRequest.incrementAndGet() % slavesCount; if(index < 0) index = - 0; DataSource ds = this.slaves.get(index); logger.debug(">>> STATUS isSlave current determine db is slaves request count {}", slaveRequest.get()); return ds; } @Override public Connection getConnection() throws SQLException { logger.debug("Enter"); return determineDataSource().getConnection(); } @Override public Connection getConnection(String username, String password) throws SQLException { return determineDataSource().getConnection(username,password); } @Override public void afterPropertiesSet() throws Exception { Assert.notNull(this.master, "property master is required"); Assert.notNull(this.slaves, "property slave/slaves is required"); this.slavesCount = this.slaves.size(); } }
相关推荐
资源内项目源码是来自个人的毕业设计,代码都测试ok,包含源码、数据集、可视化页面和部署说明,可产生核心指标曲线图、混淆矩阵、F1分数曲线、精确率-召回率曲线、验证集预测结果、标签分布图。都是运行成功后才上传资源,毕设答辩评审绝对信服的保底85分以上,放心下载使用,拿来就能用。包含源码、数据集、可视化页面和部署说明一站式服务,拿来就能用的绝对好资源!!! 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、大作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.txt文件,仅供学习参考, 切勿用于商业用途。
《基于YOLOv8的智慧社区独居老人生命体征监测系统》(包含源码、可视化界面、完整数据集、部署教程)简单部署即可运行。功能完善、操作简单,适合毕设或课程设计
Android Studio Meerkat 2024.3.1 Patch 1(android-studio-2024.3.1.14-mac.dmg)适用于macOS Intel系统,文件使用360压缩软件分割成两个压缩包,必须一起下载使用: part1: https://download.csdn.net/download/weixin_43800734/90557060 part2: https://download.csdn.net/download/weixin_43800734/90557056
侧轴承杯加工工艺编制及夹具设计.zip
NASA数据集锂电池容量特征提取(Matlab完整源码和数据) 作者介绍:机器学习之心,博客专家认证,机器学习领域创作者,2023博客之星TOP50,主做机器学习和深度学习时序、回归、分类、聚类和降维等程序设计和案例分析,文章底部有博主联系方式。从事Matlab、Python算法仿真工作8年,更多仿真源码、数据集定制私信。
板料折弯机液压系统设计.zip
C6150车床的设计.zip
机器学习之KNN实现手写数字
python爬虫;智能切换策略,反爬检测机制
mpls-vpn-optionA-all
56tgyhujikolp[
GB 6442-86企业职工伤亡事故调查分析规则.pdf
汽车液压式主动悬架系统的设计().zip
2000-2024年各省专利侵权案件结案数数据 1、时间:2000-2024年 2、来源:国家知识产权J 3、指标:专利侵权案件结案数 4、范围:31省 5、用途:可用于衡量知识产权保护水平
资源内项目源码是来自个人的毕业设计,代码都测试ok,包含源码、数据集、可视化页面和部署说明,可产生核心指标曲线图、混淆矩阵、F1分数曲线、精确率-召回率曲线、验证集预测结果、标签分布图。都是运行成功后才上传资源,毕设答辩评审绝对信服的保底85分以上,放心下载使用,拿来就能用。包含源码、数据集、可视化页面和部署说明一站式服务,拿来就能用的绝对好资源!!! 项目备注 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用! 2、本项目适合计算机相关专业(如计科、人工智能、通信工程、自动化、电子信息等)的在校学生、老师或者企业员工下载学习,也适合小白学习进阶,当然也可作为毕设项目、课程设计、大作业、项目初期立项演示等。 3、如果基础还行,也可在此代码基础上进行修改,以实现其他功能,也可用于毕设、课设、作业等。 下载后请首先打开README.txt文件,仅供学习参考, 切勿用于商业用途。
内容概要:本文档详细复现了金融数学课程作业,涵盖欧式看涨期权定价和投资组合优化两大部分。对于欧式看涨期权定价,分别采用Black-Scholes模型和蒙特卡洛方法进行了计算,并对彩虹期权进行了基于最大值的看涨期权定价。投资组合优化部分则探讨了最小方差组合、给定收益的最小方差组合、最大效用组合以及给定风险的最大收益组合四种情形,还对比了拉格朗日乘数法和二次规划求解器两种方法。文中不仅提供了详细的MATLAB代码,还有详尽的中文解释,确保每一步骤清晰明了。 适合人群:金融工程专业学生、量化分析师、金融数学爱好者。 使用场景及目标:①帮助学生理解和掌握金融衍生品定价的基本原理和方法;②为从事量化分析的专业人士提供实用工具和技术支持;③作为教学材料辅助高校教师讲授相关内容。 其他说明:文档还包括了完整的论文结构建议,从封面页到结论,再到附录,涵盖了所有必要元素,确保提交的作业符合学术规范。此外,还特别强调了数据预处理步骤,确保代码可以顺利运行。
脉冲电解射流加工喷射装置设计(1)
ThinkPad S1 (2nd Generation) 和ThinkPad Yoga 260 用户指南V3.0,包含如何拆机更换硬件
charles描述文件下载
python代码-使用人类对话数据集lora微调deepseek