springboot MongoDB 事务

本文介绍如何在SpringBoot环境中整合MongoDB副本集事务,并详细解释了版本要求、配置步骤及注意事项,包括如何处理副本集事务限制。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

有玩过mongodb的朋友大概会知道mongodb4.0版本已经可以支持多文档副本集事务。而最新版本4.2更是支持分片事务,即真正的支持分布式事务。不过当时我使用mongodb,其最新版本为4.10,4.2版本还没发布,因此本文还是以4.0版本的副本集事务来讲解。

事务整合

1、使用事务的前置条件

  • mongodb版本大于等于4,本文mongodb版本为4.10版本
  • mongodb搭建了副本集,本文mongodb为一主两从
  • 本文使用的springboot版本为springboot2+版本,具体说是springboot2.1.6版本

2、pom.xml引入

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

3、application.yml配置如下内容

spring:
  data:
    mongodb:
      uri: mongodb://root:123456@localhost:27017,localhost:27018,lcoalhost:27017
      database: springboot-learning
      authentication-database: admin
      transactionEnabled: true
      #field-naming-strategy: org.springframework.data.mapping.model.SnakeCaseFieldNamingStrategy
      #驼峰转下划线
      property-naming-strategy: SNAKE_CASE
  • 其中authentication-database为需要认证授权的数据库。
  • database为业务数据库。
  • transactionEnabled为自定义是否开启事务,为什么需要这个属性后边会说明。
  • property-naming-strategy也是自定义属性,用来用来映射当代码的属性是驼峰比如userName,映射到mongodb就为user_name,其实spring-data-common就有提供一个叫做field-naming-strategy就是用来实现类似功能,那为啥我重新造了一个轮子。那是因为当时我使用了没有达到我想要的效果

4、实现事务代码逻辑

@Bean
  @ConditionalOnProperty(name="spring.data.mongodb.transactionEnabled",havingValue = "true")
  MongoTransactionManager transactionManager(MongoDbFactory factory){

    return new MongoTransactionManager(factory);
  }

现在来讲讲为啥使用transactionEnabled这个自定义属性,当时是因为我所在项目组,其线上环境mongodb还不是副本集,如果直接加上面的代码,而mongodb不是副本集,则进行crud时,会报错,因此加了这个属性本质就是为了兼容

5、在需要使用事务的类或者方法上加上@Transactional注解

@Transactional
  public UserDTO saveUser(UserDTO userDTO) {

    User user = getUser(userDTO);

    User dbUser = userDao.saveUser(user);
    if(ValidateTransaction.YES.getValue().equals(userDTO.getValidateRollBack())) {
      throw new RuntimeException("验证事务回滚");
    }
    UserDTO dbUserDTO = modelMapper.map(dbUser,UserDTO.class);
    dbUserDTO.setGender(dbUser.getGender().getValue());



    return dbUserDTO;
  }

springboot整合mongodb副本集事务存在的坑点

副本集事务只对已经存在的mongodb中的集合起作用,如果要进行操作的集合,在mongodb中还没有,必须得先创建该集合,否则当该集合进行插入操作时,会报类似“Cannot create namespace sampledb_200.demo in multi-document transaction ”的错误

总结

通过这个例子,给我的最大的感受是springboot确实是一个神器,仅仅通过简单的几行代码就可以实现mongodb事务。而如果不使用springboot来整合,直接使用mongodb官方提供的api,就要写一坨代码。还有一点,mongodb4.2之前版本的副本集事务存在最大修改16MB、事务执行时间不能过长的限制,而这些问题在4.2版本已经得到解决,感兴趣的朋友,可以跟进下

demo链接

https://github.com/lyb-geek/springboot-learning/tree/master/springboot-mongodb

### Spring Boot 中 MongoDB事务管理 在 Spring Data MongoDB 中支持多文档事务,这使得可以在单个操作中处理多个数据库更新。为了启用此功能,在应用程序属性文件 `application.properties` 或者 `application.yml` 文件中设置连接到 replica set 而不是 standalone 实例[^3]。 对于基于 Spring Boot 应用程序中的 MongoDB 事务管理,需要引入依赖项并配置适当的支持: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-mongodb</artifactId> </dependency> ``` 定义一个带有 `@EnableMongoRepositories` 注解的配置类来开启仓库接口扫描以及声明要使用的存储库包路径[^4]: ```java @Configuration @EnableMongoRepositories(basePackages = "com.example.repository") public class MongoConfig { } ``` 创建服务层组件用于执行业务逻辑,并通过编程方式控制事务边界。下面的例子展示了如何在一个方法上应用 `@Transactional` 注释以确保所有涉及的操作都在同一个事务上下文中运行[^5]。 ```java @Service @Transactional("mongoTransactionManager") // 如果有多个事务管理器则指定名称 public class UserService { private final UserRepository userRepository; public UserService(UserRepository userRepository){ this.userRepository=userRepository; } @Transactional public void createUserAndAddress(User user, Address address) throws Exception{ try{ userRepository.save(user); if(true){throw new RuntimeException("Simulate error");} address.setUserId(user.getId()); addressRepository.save(address); }catch(Exception e){ throw e; } } } ``` 注意:上述代码片段假设存在名为 `userRepository` 和 `addressRepository` 的持久化对象实例变量,它们分别对应于两个不同的集合。当调用 `createUserAndAddress()` 方法时,如果抛出了未捕获异常,则会触发回滚机制;否则提交更改给数据库[^6]。 #### 配置事务管理器 为了让上面的服务能够正常工作还需要注册相应的平台事务管理器 bean: ```java @Bean(name="mongoTransactionManager") public PlatformTransactionManager transactionManager(MongoDbFactory dbFactory) { return new MongoTransactionManager(dbFactory); } ``` 以上就是关于 spring boot 使用 mongoDB 进行事务操作的一个简单介绍和例子说明。
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值