spring事务管理中,同一个Service类中非事务方法调用事务方法,事务失效解决方法(3种)

在平时开发中,同一个Service类中非事务方法调用事务方法,事务会失效失效,这里简单解释一下原因:spring采用动态代理机制来实现事务控制,而动态代理最终都是要调用原始对象的,而原始对象在去调用方法时,是不会再触发代理了!可以理解为同一个类中非事务方法调用方法时用的是当前对象去调用,而不是spring生成的代理对象,所以会导致事务失效。

演示一下事务失效:

@Service
public class UserServiceImpl implements UserService {

	@Autowired
	private UserMapper userMapper;

	@Override
	@Transactional(readOnly = true)
	public List<User> find() throws Exception {
		return userMapper.selectList(null);
	}

	@Override
	@Transactional(rollbackFor = Exception.class)
	public int saveUser(User user) throws Exception {
		int result = userMapper.insert(user);
		result = 1 / 0;
		return result;
	}

	/**
	 * 非事务方法
	 */
	@Override
	public int doSomething() throws Exception {
		User user = new User();
		user.setName("张三");
		user.setAge(23);
		user.setCreateTime(LocalDateTime.now());
		return saveUser(user);
	}

}

调用doSomething()后,日志打印:

DEBUG==>  Preparing: INSERT INTO t_user ( id, name, age, create_time ) VALUES ( ?, ?, ?, ? ) 
DEBUG==> Parameters: 1255426392998391809(Long), 张三(String), 23(Integer), 2020-04-29T17:19:11.418(LocalDateTime)
DEBUG<==    Updates: 1
java.lang.ArithmeticException: / by zero
	at com.learn.service.impl.UserServiceImpl.saveUser(UserServiceImpl.java:30)

数据库:
在这里插入图片描述
可以看到虽然程序出现了异常,可是数据库却依然保存了数据,并没有回滚事务。原因开头已经说过了,下面说一下解决方法:

解决方法:

1.将需要进行事务管理的方法单独写到一个Service文件中:

@Service
public class UserSaveService {
	@Autowired
	private UserMapper userMapper;

	@Transactional(rollbackFor = Exception.class)
	public int saveUser(User user) throws Exception {
		int result = userMapper.insert(user);
		result = 1 / 0;
		return result;
	}
}
@Service
public class UserServiceImpl implements UserService {
	@Autowired
	private UserSaveService userSaveService; // 注入另一个Service对象

	/**
	 * 非事务方法
	 */
	@Override
	public int doSomething() throws Exception {
		User user = new User();
		user.setName("李四");
		user.setAge(23);
		user.setCreateTime(LocalDateTime.now());
		return userSaveService.saveUser(user);
	}

}

异常结果:

DEBUG==>  Preparing: INSERT INTO t_user ( id, name, age, create_time ) VALUES ( ?, ?, ?, ? ) 
DEBUG==> Parameters: 1255428835366748161(Long), 李四(String), 23(Integer), 2020-04-29T17:28:52.789(LocalDateTime)
DEBUG<==    Updates: 1
java.lang.ArithmeticException: / by zero
	at com.learn.service.impl.UserSaveService.saveUser(UserSaveService.java:18)

数据库:
在这里插入图片描述
李四没有保存进数据库,事务控制成功!

2.使用 AopContext.currentProxy() 获取代理对象:

增加maven依赖:

		<dependency>
			<groupId>org.springframework.boot</groupId>
			<artifactId>spring-boot-starter-aop</artifactId>
		</dependency>

springboot启动类加上注解:@EnableAspectJAutoProxy(exposeProxy = true)

@SpringBootApplication
@MapperScan(basePackages = "com.learn.dao")
@EnableTransactionManagement
@EnableAspectJAutoProxy(exposeProxy = true)
public class SpringbootMybatisplusApplication {

	public static void main(String[] args) {
		SpringApplication.run(SpringbootMybatisplusApplication.class, args);
	}
}

Service类:

@Service
public class UserServiceImpl implements UserService {
	@Autowired
	private UserMapper userMapper;

	@Transactional(rollbackFor = Exception.class)
	public int saveUser(User user) throws Exception {
		int result = userMapper.insert(user);
		result = 1 / 0;
		return result;
	}

	/**
	 * 非事务方法
	 */
	@Override
	public int doSomething() throws Exception {
		User user = new User();
		user.setName("王五");
		user.setAge(23);
		user.setCreateTime(LocalDateTime.now());
		UserServiceImpl currentProxy = (UserServiceImpl) AopContext.currentProxy(); // 获取代理对象
		return currentProxy.saveUser(user);
	}

}

异常信息:

DEBUG==>  Preparing: INSERT INTO t_user ( id, name, age, create_time ) VALUES ( ?, ?, ?, ? ) 
DEBUG==> Parameters: 1255432158203445249(Long), 王五(String), 23(Integer), 2020-04-29T17:42:05.152(LocalDateTime)
DEBUG<==    Updates: 1
java.lang.ArithmeticException: / by zero
	at com.learn.service.impl.UserServiceImpl.saveUser(UserServiceImpl.java:22)

数据库:
在这里插入图片描述
王五没有保存进数据库,事务控制成功!

3.将该类对象在其他类中维护成一个成员变量,借助这个成员变量调用方法:

例如构造一个工具类:

@Component
public class ServiceUtil {

	private static ServiceUtil serviceUtil;

	@Autowired
	private UserService userService;

	@PostConstruct
	public void init() {
		serviceUtil = this;
		serviceUtil.userService = this.userService;
	}

	public static UserService getUserService() {
		return serviceUtil.userService;
	}

}

使用工具类调用方法:

@Service
public class UserServiceImpl implements UserService {
	@Autowired
	private UserMapper userMapper;

	@Transactional(rollbackFor = Exception.class)
	public int saveUser(User user) throws Exception {
		int result = userMapper.insert(user);
		result = 1 / 0;
		return result;
	}

	/**
	 * 非事务方法
	 */
	@Override
	public int doSomething() throws Exception {
		User user = new User();
		user.setName("赵六");
		user.setAge(23);
		user.setCreateTime(LocalDateTime.now());
		// 使用工具类调用方法
		return ServiceUtil.getUserService().saveUser(user); 
	}

}

异常信息:

DEBUG==>  Preparing: INSERT INTO t_user ( id, name, age, create_time ) VALUES ( ?, ?, ?, ? ) 
DEBUG==> Parameters: 1255665465251971073(Long), 赵六(String), 23(Integer), 2020-04-30T09:09:10.708(LocalDateTime)
DEBUG<==    Updates: 1
java.lang.ArithmeticException: / by zero
	at com.learn.service.impl.UserServiceImpl.saveUser(UserServiceImpl.java:22)

数据库:
在这里插入图片描述
赵六没有保存进数据库,事务控制成功!

另外,你也可以在类上加注解@Transactional,这样所有方法都可以进行事务管理,但是不推荐这么做,特别是有些方法执行时间很长,这种尽量不添加事务管理,而是在里面调用的具体方法上进行事务管理,不然可能产生长事务,导致数据库连接被长期占用,导致未知异常,比如一个简单查询都不能及时执行!

  • 4
    点赞
  • 16
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
当在同一个方法相互调用时,如果希望事务能够生效,可以采取以下解决办法: 1. 使用代理调用方法:由于Spring事务管理是通过AOP代理实现的,所以可以通过使用代理对象调用方法来触发事务管理。可以通过将方法调用委托给代理对象来确保事务的生效。 2. 将被调用方法抽取到另一个:将被调用方法抽取到另一个,并确保在被调用方法上添加@Transactional注解。这样,在调用方法调用被抽取的方法时,事务将能够生效。 3. 使用AspectJ模式的事务管理Spring还提供了AspectJ模式的事务管理,可以在同一个方法相互调用时保持事务的生效。通过配置AspectJ的切面来实现事务的管理,可以细粒度地控制事务的传播行为和回滚条件。 需要注意的是,以上解决办法需要根据具体情况选择合适的方式,并确保在调用方法上正确地添加@Transactional注解。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [Spring同一个service方法相互调用事务不生效问题解决方案](https://blog.csdn.net/a1036645146/article/details/107469578)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值