spring事物提交后mysql数据没有变化

spring事物提交后mysql数据没有变化,第1张

spring事物提交后mysql数据没有变化应该用了spring配置式事务,被限制了修改了

Spring是一个开放源代码的J2EE应用程序框架,由Rod Johnson发起,是针对bean的生命周期进行管理的轻量级容器(lightweight container)。

spring事务失效场景可能大家在很多文章都看过了,所以今天就水一篇,看大家能不能收获一些不一样的东西。直接进入主题

失效原因: spring事务生效的前提是,service必须是一个bean对象

解决方案: 将service注入spring

失效原因: spring默认只会回滚非检查异常和error异常

解决方案: 配置rollbackFor

失效原因: spring事务只有捕捉到了业务抛出去的异常,才能进行后续的处理,如果业务自己捕获了异常,则事务无法感知

解决方案:

1、将异常原样抛出;

2、设置TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()

失效原因: spring事务切面的优先级顺序最低,但如果自定义的切面优先级和他一样,且自定义的切面没有正确处理异常,则会同业务自己捕获异常的那种场景一样

解决方案:

1、在切面中将异常原样抛出;

2、在切面中设置TransactionAspectSupport.currentTransactionStatus().setRollbackOnly()

失效原因: spring事务默认生效的方法权限都必须为public

解决方案:

1、将方法改为public;

2、修改TansactionAttributeSource,将publicMethodsOnly改为false【这个从源码跟踪得出结论】

3、开启 AspectJ 代理模式【从spring文档得出结论】

具体步骤:

1、在pom引入aspectjrt坐标以及相应插件

2、在启动类上加上如下配置

注: 如果是在idea上运行,则需做如下配置

4、直接用TransactionTemplate

示例:

失效原因: 子容器扫描范围过大,将未加事务配置的serivce扫描进来

解决方案:

1、父子容器个扫个的范围;

2、不用父子容器,所有bean都交给同一容器管理

注: 因为示例是使用springboot,而springboot启动默认没有父子容器,只有一个容器,因此就该场景就演示示例了

失效原因: 因为spring事务是用动态代理实现,因此如果方法使用了final修饰,则代理类无法对目标方法进行重写,植入事务功能

解决方案:

1、方法不要用final修饰

失效原因: 原因和final一样

解决方案:

1、方法不要用static修饰

失效原因: 本类方法不经过代理,无法进行增强

解决方案:

1、注入自己来调用;

2、使用@EnableAspectJAutoProxy(exposeProxy = true) + AopContext.currentProxy()

失效原因: 因为spring的事务是通过数据库连接来实现,而数据库连接spring是放在threadLocal里面。同一个事务,只能用同一个数据库连接。而多线程场景下,拿到的数据库连接是不一样的,即是属于不同事务

失效原因: 使用的传播特性不支持事务

失效原因: 使用了不支持事务的存储引擎。比如mysql中的MyISAM

注: 因为springboot,他默认已经开启事务管理器。org.springframework.boot.autoconfigure.jdbc.DataSourceTransactionManagerAutoConfiguration。因此示例略过

失效原因: 当代理类的实例化早于AbstractAutoProxyCreator后置处理器,就无法被AbstractAutoProxyCreator后置处理器增强

本文列举了14种spring事务失效的场景,其实这14种里面有很多都是归根结底都是属于同一类问题引起,比如因为动态代理原因、方法限定符原因、异常类型原因等

https://github.com/lyb-geek/springboot-learning/tree/master/springboot-transaction-invalid-case

可以给所有读的方法添加一个参数,来控制读从库还是主库。

2、数据源如何路由?

spring-jdbc 包中提供了一个抽象类:AbstractRoutingDataSource,实现了javax.sql.DataSource接口,我们用这个类来作为数据源类,重点是这个类可以用来做数据源的路由,可以在其内部配置多个真实的数据源,最终用哪个数据源,由开发者来决定。

AbstractRoutingDataSource中有个map,用来存储多个目标数据源

private Map<Object, DataSource>resolvedDataSources

比如主从库可以这么存储

resolvedDataSources.put("master",主库数据源)

resolvedDataSources.put("salave",从库数据源)

AbstractRoutingDataSource中还有抽象方法determineCurrentLookupKey,将这个方法的返回值作为key到上面的resolvedDataSources中查找对应的数据源,作为当前 *** 作db的数据源

protected abstract Object determineCurrentLookupKey()

3、读写分离在哪控制?

读写分离属于一个通用的功能,可以通过spring的aop来实现,添加一个拦截器,拦截目标方法的之前,在目标方法执行之前,获取一下当前需要走哪个库,将这个标志存储在ThreadLocal中,将这个标志作为AbstractRoutingDataSource.determineCurrentLookupKey()方法的返回值,拦截器中在目标方法执行完毕之后,将这个标志从ThreadLocal中清除。

3、代码实现

3.1、工程结构图

3.2、DsType

表示数据源类型,有2个值,用来区分是主库还是从库。

package com.javacode2018.readwritesplit.base

public enum DsType {

MASTER, SLAVE

}

3.3、DsTypeHolder

内部有个ThreadLocal,用来记录当前走主库还是从库,将这个标志放在dsTypeThreadLocal中

package com.javacode2018.readwritesplit.base

public class DsTypeHolder {

private static ThreadLocal<DsType>dsTypeThreadLocal = new ThreadLocal<>()

public static void master() {

dsTypeThreadLocal.set(DsType.MASTER)

}

public static void slave() {

dsTypeThreadLocal.set(DsType.SLAVE)

}

public static DsType getDsType() {

return dsTypeThreadLocal.get()

}

public static void clearDsType() {

dsTypeThreadLocal.remove()

}

}

3.4、IService接口

这个接口起到标志的作用,当某个类需要启用读写分离的时候,需要实现这个接口,实现这个接口的类都会被读写分离拦截器拦截。

package com.javacode2018.readwritesplit.base

//需要实现读写分离的service需要实现该接口

public interface IService {

}

3.5、ReadWriteDataSource

读写分离数据源,继承ReadWriteDataSource,注意其内部的determineCurrentLookupKey方法,从上面的ThreadLocal中获取当前需要走主库还是从库的标志。

package com.javacode2018.readwritesplit.base

import org.springframework.jdbc.datasource.lookup.AbstractRoutingDataSource

import org.springframework.lang.Nullable

public class ReadWriteDataSource extends AbstractRoutingDataSource {

@Nullable

@Override

protected Object determineCurrentLookupKey() {

return DsTypeHolder.getDsType()

}

}

3.6、ReadWriteInterceptor

读写分离拦截器,需放在事务拦截器前面执行,通过@1代码我们将此拦截器的顺序设置为Integer.MAX_VALUE - 2,稍后我们将事务拦截器的顺序设置为Integer.MAX_VALUE - 1,事务拦截器的执行顺序是从小到达的,所以,ReadWriteInterceptor会在事务拦截器org.springframework.transaction.interceptor.TransactionInterceptor之前执行。

由于业务方法中存在相互调用的情况,比如service1.m1中调用service2.m2,而service2.m2中调用了service2.m3,我们只需要在m1方法执行之前,获取具体要用哪个数据源就可以了,所以下面代码中会在第一次进入这个拦截器的时候,记录一下走主库还是从库。

下面方法中会获取当前目标方法的最后一个参数,最后一个参数可以是DsType类型的,开发者可以通过这个参数来控制具体走主库还是从库。

package com.javacode2018.readwritesplit.base

import org.aspectj.lang.ProceedingJoinPoint

import org.aspectj.lang.annotation.Around

import org.aspectj.lang.annotation.Aspect

import org.aspectj.lang.annotation.Pointcut

import org.springframework.core.annotation.Order

import org.springframework.stereotype.Component

import java.util.Objects

@Aspect

@Order(Integer.MAX_VALUE - 2) //@1

@Component

public class ReadWriteInterceptor {

@Pointcut("target(IService)")

public void pointcut() {

}

//获取当前目标方法的最后一个参数

private Object getLastArgs(final ProceedingJoinPoint pjp) {

Object[] args = pjp.getArgs()

if (Objects.nonNull(args) &&args.length >0) {

return args[args.length - 1]

} else {

return null

}

}

@Around("pointcut()")

public Object around(final ProceedingJoinPoint pjp) throws Throwable {

//判断是否是第一次进来,用于处理事务嵌套

boolean isFirst = false

try {

if (DsTypeHolder.getDsType() == null) {

isFirst = true

}

if (isFirst) {

Object lastArgs = getLastArgs(pjp)

if (DsType.SLAVE.equals(lastArgs)) {

DsTypeHolder.slave()

} else {

DsTypeHolder.master()

}

}

return pjp.proceed()

} finally {

//退出的时候,清理

if (isFirst) {

DsTypeHolder.clearDsType()

}

}

}

}

3.7、ReadWriteConfiguration

spring配置类,作用

1、@3:用来将com.javacode2018.readwritesplit.base包中的一些类注册到spring容器中,比如上面的拦截器ReadWriteInterceptor

2、@1:开启spring aop的功能

3、@2:开启spring自动管理事务的功能,@EnableTransactionManagement的order用来指定事务拦截器org.springframework.transaction.interceptor.TransactionInterceptor顺序,在这里我们将order设置为Integer.MAX_VALUE - 1,而上面ReadWriteInterceptor的order是Integer.MAX_VALUE - 2,所以ReadWriteInterceptor会在事务拦截器之前执行。

package com.javacode2018.readwritesplit.base

import org.springframework.context.annotation.ComponentScan

import org.springframework.context.annotation.Configuration

import org.springframework.context.annotation.EnableAspectJAutoProxy

import org.springframework.transaction.annotation.EnableTransactionManagement

@Configuration

@EnableAspectJAutoProxy //@1

@EnableTransactionManagement(proxyTargetClass = true, order = Integer.MAX_VALUE - 1) //@2

@ComponentScan(basePackageClasses = IService.class) //@3

public class ReadWriteConfiguration {

}

3.8、@EnableReadWrite

这个注解用来开启读写分离的功能,@1通过@Import将ReadWriteConfiguration导入到spring容器了,这样就会自动启用读写分离的功能。业务中需要使用读写分离,只需要在spring配置类中加上@EnableReadWrite注解就可以了。

package com.javacode2018.readwritesplit.base

import org.springframework.context.annotation.Import

import java.lang.annotation.*

@Target(ElementType.TYPE)

@Retention(RetentionPolicy.RUNTIME)

@Documented

@Import(ReadWriteConfiguration.class) //@1

public @interface EnableReadWrite {

}

4、案例

读写分离的关键代码写完了,下面我们来上案例验证一下效果。

4.1、执行sql脚本

下面准备2个数据库:javacode2018_master(主库)、javacode2018_slave(从库)

2个库中都创建一个t_user表,分别插入了一条数据,稍后用这个数据来验证走的是主库还是从库。

DROP DATABASE IF EXISTS javacode2018_master

CREATE DATABASE IF NOT EXISTS javacode2018_master

USE javacode2018_master

DROP TABLE IF EXISTS t_user

CREATE TABLE t_user (

id INT PRIMARY KEY AUTO_INCREMENT,

name VARCHAR(256) NOT NULL DEFAULT ''

COMMENT '姓名'

)

INSERT INTO t_user (name) VALUE ('master库')

DROP DATABASE IF EXISTS javacode2018_slave

CREATE DATABASE IF NOT EXISTS javacode2018_slave

USE javacode2018_slave

DROP TABLE IF EXISTS t_user

CREATE TABLE t_user (

id INT PRIMARY KEY AUTO_INCREMENT,

name VARCHAR(256) NOT NULL DEFAULT ''

COMMENT '姓名'

)

INSERT INTO t_user (name) VALUE ('slave库')

4.2、spring配置类

@1:启用读写分离

masterDs()方法:定义主库数据源

slaveDs()方法:定义从库数据源

dataSource():定义读写分离路由数据源

后面还有2个方法用来定义JdbcTemplate和事务管理器,方法中都通过@Qualifier(“dataSource”)限定了注入的bean名称为dataSource:即注入了上面dataSource()返回的读写分离路由数据源。

package com.javacode2018.readwritesplit.demo1

import com.javacode2018.readwritesplit.base.DsType

import com.javacode2018.readwritesplit.base.EnableReadWrite

import com.javacode2018.readwritesplit.base.ReadWriteDataSource

import org.springframework.beans.factory.annotation.Qualifier

import org.springframework.context.annotation.Bean

import org.springframework.context.annotation.ComponentScan

import org.springframework.context.annotation.Configuration

import org.springframework.jdbc.core.JdbcTemplate

import org.springframework.jdbc.datasource.DataSourceTransactionManager

import org.springframework.transaction.PlatformTransactionManager

import javax.sql.DataSource

import java.util.HashMap

import java.util.Map

@EnableReadWrite //@1

@Configuration

@ComponentScan

public class MainConfig {

//主库数据源

@Bean

public DataSource masterDs() {

org.apache.tomcat.jdbc.pool.DataSource dataSource = new org.apache.tomcat.jdbc.pool.DataSource()

dataSource.setDriverClassName("com.mysql.jdbc.Driver")

dataSource.setUrl("jdbc:mysql://localhost:3306/javacode2018_master?characterEncoding=UTF-8")

dataSource.setUsername("root")

dataSource.setPassword("root123")

dataSource.setInitialSize(5)

return dataSource

}

//从库数据源

@Bean

public DataSource slaveDs() {

org.apache.tomcat.jdbc.pool.DataSource dataSource = new org.apache.tomcat.jdbc.pool.DataSource()

dataSource.setDriverClassName("com.mysql.jdbc.Driver")

dataSource.setUrl("jdbc:mysql://localhost:3306/javacode2018_slave?characterEncoding=UTF-8")

dataSource.setUsername("root")

dataSource.setPassword("root123")

dataSource.setInitialSize(5)

return dataSource

}

//读写分离路由数据源

@Bean

public ReadWriteDataSource dataSource() {

ReadWriteDataSource dataSource = new ReadWriteDataSource()

//设置主库为默认的库,当路由的时候没有在datasource那个map中找到对应的数据源的时候,会使用这个默认的数据源

dataSource.setDefaultTargetDataSource(this.masterDs())

//设置多个目标库

Map<Object, Object>targetDataSources = new HashMap<>()

targetDataSources.put(DsType.MASTER, this.masterDs())

targetDataSources.put(DsType.SLAVE, this.slaveDs())

dataSource.setTargetDataSources(targetDataSources)

return dataSource

}

//JdbcTemplate,dataSource为上面定义的注入读写分离的数据源

@Bean

public JdbcTemplate jdbcTemplate(@Qualifier("dataSource") DataSource dataSource) {

return new JdbcTemplate(dataSource)

}

//定义事务管理器,dataSource为上面定义的注入读写分离的数据源

@Bean

public PlatformTransactionManager transactionManager(@Qualifier("dataSource") DataSource dataSource) {

return new DataSourceTransactionManager(dataSource)

}

}

4.3、UserService

这个类就相当于我们平时写的service,我是为了方法,直接在里面使用了JdbcTemplate来 *** 作数据库,真实的项目 *** 作db会放在dao里面。

getUserNameById方法:通过id查询name。

insert方法:插入数据,这个内部的所有 *** 作都会走主库,为了验证是不是查询也会走主库,插入数据之后,我们会调用this.userService.getUserNameById(id, DsType.SLAVE)方法去执行查询 *** 作,第二个参数故意使用SLAVE,如果查询有结果,说明走的是主库,否则走的是从库,这里为什么需要通过this.userService来调用getUserNameById?

this.userService最终是个代理对象,通过代理对象访问其内部的方法,才会被读写分离的拦截器拦截。

package com.javacode2018.readwritesplit.demo1

import com.javacode2018.readwritesplit.base.DsType

import com.javacode2018.readwritesplit.base.IService

import org.springframework.beans.factory.annotation.Autowired

import org.springframework.jdbc.core.JdbcTemplate

import org.springframework.stereotype.Component

import org.springframework.transaction.annotation.Propagation

import org.springframework.transaction.annotation.Transactional

import java.util.List

@Component

public class UserService implements IService {

@Autowired

private JdbcTemplate jdbcTemplate

@Autowired

private UserService userService

@Transactional(propagation = Propagation.SUPPORTS, readOnly = true)

public String getUserNameById(long id, DsType dsType) {

String sql = "select name from t_user where id=?"

List<String>list = this.jdbcTemplate.queryForList(sql, String.class, id)

return (list != null &&list.size() >0) ? list.get(0) : null

}

//这个insert方法会走主库,内部的所有 *** 作都会走主库

@Transactional

public void insert(long id, String name) {

System.out.println(String.format("插入数据{id:%s, name:%s}", id, name))

this.jdbcTemplate.update("insert into t_user (id,name) values (?,?)", id, name)

String userName = this.userService.getUserNameById(id, DsType.SLAVE)

System.out.println("查询结果:" + userName)

}

}

4.4、测试用例

package com.javacode2018.readwritesplit.demo1

import com.javacode2018.readwritesplit.base.DsType

import org.junit.Before

import org.junit.Test

import org.springframework.context.annotation.AnnotationConfigApplicationContext

public class Demo1Test {

UserService userService

@Before

public void before() {

AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext()

context.register(MainConfig.class)

context.refresh()

this.userService = context.getBean(UserService.class)

}

@Test

public void test1() {

System.out.println(this.userService.getUserNameById(1, DsType.MASTER))

System.out.println(this.userService.getUserNameById(1, DsType.SLAVE))

}

@Test

public void test2() {

long id = System.currentTimeMillis()

System.out.println(id)

this.userService.insert(id, "张三")

}

}

test1方法执行2次查询,分别查询主库和从库,输出:

master库

slave库

是不是很爽,由开发者自己控制具体走主库还是从库。

test2执行结果如下,可以看出查询到了刚刚插入的数据,说明insert中所有 *** 作都走的是主库。

1604905117467

插入数据{id:1604905117467, name:张三}

查询结果:张三

5、案例源码


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/zaji/6096589.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-03-14
下一篇 2023-03-14

发表评论

登录后才能评论

评论列表(0条)

保存