springboot学习笔记(九)mongo多数据源配置

项目当中有时候一个业务需要用到多个数据源,但是又不是微服务的形式,要在一个项目当中配置多个mongo数据源,如何配置呢?

1.springboot 资源文件(两个mongo地址):

spring.data.mongodb.primary.uri=mongodb://192.9.8.204:27017/reslib_server
spring.data.mongodb.secondary.uri=mongodb://192.9.8.204:27017/cloudfsdb

2.写配置类

package com.tyky.educloud.portal.config;

import org.springframework.boot.context.properties.ConfigurationProperties;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Primary;
import org.springframework.data.mongodb.MongoDbFactory;
import org.springframework.data.mongodb.core.MongoTemplate;
import org.springframework.data.mongodb.core.SimpleMongoDbFactory;
import org.springframework.data.mongodb.repository.config.EnableMongoRepositories;

import com.mongodb.MongoClientURI;


@Configuration  //Configuration class  
@ConfigurationProperties(prefix = "spring.data.mongodb.primary") 
@EnableMongoRepositories(basePackages="com.tyky.educloud.portal.repository",mongoTemplateRef="mongoTemplate")
public class PrimaryMongoConfig {
  private String uri;     
  
  @Primary      
  @Bean(name="mongoTemplate")
  public MongoTemplate getMongoTemplate() throws Exception {          
      return new MongoTemplate(mongoDbFactory());      
  }
  
  @Primary
  @Bean(name="mongoDbFactory")
  public MongoDbFactory mongoDbFactory() throws Exception {
      MongoClientURI mongoClientURI = new MongoClientURI(uri);      
      return new SimpleMongoDbFactory(mongoClientURI);
  }

  public String getUri() {
    return uri;
  }

  public void setUri(String uri) {
    this.uri = uri;
  }


}

其中

@EnableMongoRepositories(basePackages="com.tyky.educloud.portal.repository",mongoTemplateRef="mongoTemplate")

指定到你这个数据源对应的仓储类。


第二个数据源配置类:

package com.tyky.educloud.portal.config;

import java.util.Collections;
import java.util.HashSet;
import java.util.Set;

import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.boot.context.properties.ConfigurationProperties;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.convert.converter.Converter;
import org.springframework.core.type.filter.AnnotationTypeFilter;
import org.springframework.data.annotation.Persistent;
import org.springframework.data.authentication.UserCredentials;
import org.springframework.data.mapping.context.MappingContextIsNewStrategyFactory;
import org.springframework.data.mapping.model.CamelCaseAbbreviatingFieldNamingStrategy;
import org.springframework.data.mapping.model.FieldNamingStrategy;
import org.springframework.data.mapping.model.PropertyNameFieldNamingStrategy;
import org.springframework.data.mongodb.MongoDbFactory;
import org.springframework.data.mongodb.config.AbstractMongoConfiguration;
import org.springframework.data.mongodb.core.MongoTemplate;
import org.springframework.data.mongodb.core.SimpleMongoDbFactory;
import org.springframework.data.mongodb.core.convert.CustomConversions;
import org.springframework.data.mongodb.core.convert.DbRefResolver;
import org.springframework.data.mongodb.core.convert.DefaultDbRefResolver;
import org.springframework.data.mongodb.core.convert.MappingMongoConverter;
import org.springframework.data.mongodb.core.mapping.Document;
import org.springframework.data.mongodb.core.mapping.MongoMappingContext;
import org.springframework.data.mongodb.gridfs.GridFsTemplate;
import org.springframework.data.support.CachingIsNewStrategyFactory;
import org.springframework.data.support.IsNewStrategyFactory;
import org.springframework.util.ClassUtils;
import org.springframework.util.StringUtils;

import com.mongodb.MongoClient;
import com.mongodb.MongoClientURI;


@Configuration  //Configuration class  
@ConfigurationProperties(prefix = "spring.data.mongodb.secondary") 
public class SecondartyMongoConfig {
  private String uri;   
  
  @Bean(name="cloudfsTemplate")
  public MongoTemplate getMongoTemplate() throws Exception {          
      return new MongoTemplate(mongoDbFactory());      
  }
  
  @Bean(name="secondaryMongoDbFactory")
  public MongoDbFactory mongoDbFactory() throws Exception {
      MongoClientURI mongoClientURI = new MongoClientURI(uri);      
      return new SimpleMongoDbFactory(mongoClientURI);
  }
  
  @Bean(name="cloudfsgrid")
  public GridFsTemplate getGridFsTemplate() throws Exception {          
      return new GridFsTemplate(mongoDbFactory(), mappingMongoConverter());   
  }
 

  /**
   * Return the base package to scan for mapped {@link Document}s. Will return the package name of the configuration
   * class' (the concrete class, not this one here) by default. So if you have a {@code com.acme.AppConfig} extending
   * {@link AbstractMongoConfiguration} the base package will be considered {@code com.acme} unless the method is
   * overriden to implement alternate behaviour.
   * 
   * @return the base package to scan for mapped {@link Document} classes or {@literal null} to not enable scanning for
   *         entities.
   */
  protected String getMappingBasePackage() {

      Package mappingBasePackage = getClass().getPackage();
      return mappingBasePackage == null ? null : mappingBasePackage.getName();
  }

  /**
   * Return {@link UserCredentials} to be used when connecting to the MongoDB instance or {@literal null} if none shall
   * be used.
   * 
   * @return
   * @deprecated since 1.7. {@link MongoClient} should hold authentication data within
   *             {@link MongoClient#getCredentialsList()}
   */
  @Deprecated
  protected UserCredentials getUserCredentials() {
      return null;
  }

  /**
   * Creates a {@link MongoMappingContext} equipped with entity classes scanned from the mapping base package.
   * 
   * @see #getMappingBasePackage()
   * @return
   * @throws ClassNotFoundException
   */
  @Bean
  public MongoMappingContext mongoMappingContext() throws ClassNotFoundException {

      MongoMappingContext mappingContext = new MongoMappingContext();
      mappingContext.setInitialEntitySet(getInitialEntitySet());
      mappingContext.setSimpleTypeHolder(customConversions().getSimpleTypeHolder());
      mappingContext.setFieldNamingStrategy(fieldNamingStrategy());

      return mappingContext;
  }

  /**
   * Returns a {@link MappingContextIsNewStrategyFactory} wrapped into a {@link CachingIsNewStrategyFactory}.
   * 
   * @return
   * @throws ClassNotFoundException
   */
  @Bean
  public IsNewStrategyFactory isNewStrategyFactory() throws ClassNotFoundException {
      return new CachingIsNewStrategyFactory(new MappingContextIsNewStrategyFactory(mongoMappingContext()));
  }

  /**
   * Register custom {@link Converter}s in a {@link CustomConversions} object if required. These
   * {@link CustomConversions} will be registered with the {@link #mappingMongoConverter()} and
   * {@link #mongoMappingContext()}. Returns an empty {@link CustomConversions} instance by default.
   * 
   * @return must not be {@literal null}.
   */
  @Bean
  public CustomConversions customConversions() {
      return new CustomConversions(Collections.emptyList());
  }

  /**
   * Creates a {@link MappingMongoConverter} using the configured {@link #mongoDbFactory()} and
   * {@link #mongoMappingContext()}. Will get {@link #customConversions()} applied.
   * 
   * @see #customConversions()
   * @see #mongoMappingContext()
   * @see #mongoDbFactory()
   * @return
   * @throws Exception
   */
  @Bean
  public MappingMongoConverter mappingMongoConverter() throws Exception {

      DbRefResolver dbRefResolver = new DefaultDbRefResolver(mongoDbFactory());
      MappingMongoConverter converter = new MappingMongoConverter(dbRefResolver, mongoMappingContext());
      converter.setCustomConversions(customConversions());

      return converter;
  }

  /**
   * Scans the mapping base package for classes annotated with {@link Document}.
   * 
   * @see #getMappingBasePackage()
   * @return
   * @throws ClassNotFoundException
   */
  protected Set<Class<?>> getInitialEntitySet() throws ClassNotFoundException {

      String basePackage = getMappingBasePackage();
      Set<Class<?>> initialEntitySet = new HashSet<Class<?>>();

      if (StringUtils.hasText(basePackage)) {
          ClassPathScanningCandidateComponentProvider componentProvider = new ClassPathScanningCandidateComponentProvider(
                  false);
          componentProvider.addIncludeFilter(new AnnotationTypeFilter(Document.class));
          componentProvider.addIncludeFilter(new AnnotationTypeFilter(Persistent.class));

          for (BeanDefinition candidate : componentProvider.findCandidateComponents(basePackage)) {
              initialEntitySet.add(ClassUtils.forName(candidate.getBeanClassName(),
                      AbstractMongoConfiguration.class.getClassLoader()));
          }
      }

      return initialEntitySet;
  }

  /**
   * Configures whether to abbreviate field names for domain objects by configuring a
   * {@link CamelCaseAbbreviatingFieldNamingStrategy} on the {@link MongoMappingContext} instance created. For advanced
   * customization needs, consider overriding {@link #mappingMongoConverter()}.
   * 
   * @return
   */
  protected boolean abbreviateFieldNames() {
      return false;
  }

  /**
   * Configures a {@link FieldNamingStrategy} on the {@link MongoMappingContext} instance created.
   * 
   * @return
   * @since 1.5
   */
  protected FieldNamingStrategy fieldNamingStrategy() {
      return abbreviateFieldNames() ? new CamelCaseAbbreviatingFieldNamingStrategy()
              : PropertyNameFieldNamingStrategy.INSTANCE;
  }

  public String getUri() {
    return uri;
  }

  public void setUri(String uri) {
    this.uri = uri;
  }
  
}

3.业务当中的使用:

	@Resource
	@Qualifier("mongoTemplate")
	private MongoOperations mongoOps;

	@Resource
	@Qualifier("cloudfsgrid")
	private GridFsOperations gridFsOps;


  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Spring Boot 中整合 MongoDB 多数据源,需要按照以下步骤操作: 1. 引入 MongoDB 的依赖 在 pom.xml 文件中引入 MongoDB 的依赖: ```xml <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-mongodb</artifactId> </dependency> ``` 2. 配置数据源 在 application.properties 文件中配置多个数据源: ```properties # 数据源1 spring.data.mongodb.uri=mongodb://localhost:27017/db1 # 数据源2 mongodb2.uri=mongodb://localhost:27017/db2 ``` 3. 配置多数据源 创建多个数据源的配置类,继承自 `AbstractMongoConfiguration`,并重写 `mongoClient()` 方法: ```java @Configuration public class DataSourceConfig1 extends AbstractMongoConfiguration { @Value("${spring.data.mongodb.uri}") private String uri; @Override protected String getDatabaseName() { return "db1"; } @Override public MongoClient mongoClient() { return new MongoClient(new MongoClientURI(uri)); } } @Configuration public class DataSourceConfig2 extends AbstractMongoConfiguration { @Value("${mongodb2.uri}") private String uri; @Override protected String getDatabaseName() { return "db2"; } @Override public MongoClient mongoClient() { return new MongoClient(new MongoClientURI(uri)); } } ``` 4. 配置动态数据源 创建一个动态数据源,实现 `AbstractRoutingDataSource` 接口,重写 `determineCurrentLookupKey()` 方法: ```java public class DynamicDataSource extends AbstractRoutingDataSource { @Override protected Object determineCurrentLookupKey() { return DataSourceContextHolder.getDataSource(); } } ``` 5. 配置数据源上下文 创建一个数据源上下文,用于保存当前使用的数据源的名称: ```java public class DataSourceContextHolder { private static final ThreadLocal<String> contextHolder = new ThreadLocal<>(); public static void setDataSource(String dataSource) { contextHolder.set(dataSource); } public static String getDataSource() { return contextHolder.get(); } public static void clearDataSource() { contextHolder.remove(); } } ``` 6. 配置事务管理器 创建一个事务管理器,用于管理多个数据源的事务: ```java @Configuration @EnableTransactionManagement public class TransactionConfig implements TransactionManagementConfigurer { @Autowired private DynamicDataSource dynamicDataSource; @Override public PlatformTransactionManager annotationDrivenTransactionManager() { return new DataSourceTransactionManager(dynamicDataSource); } } ``` 7. 完成动态数据源配置 在 `Application` 类中完成动态数据源的配置: ```java @SpringBootApplication public class Application { public static void main(String[] args) { SpringApplication.run(Application.class, args); } @Bean public DynamicDataSource dynamicDataSource(DataSourceConfig1 dataSourceConfig1, DataSourceConfig2 dataSourceConfig2) { Map<Object, Object> targetDataSources = new HashMap<>(); targetDataSources.put("db1", dataSourceConfig1.mongoClient()); targetDataSources.put("db2", dataSourceConfig2.mongoClient()); DynamicDataSource dynamicDataSource = new DynamicDataSource(); dynamicDataSource.setDefaultTargetDataSource(dataSourceConfig1.mongoClient()); dynamicDataSource.setTargetDataSources(targetDataSources); return dynamicDataSource; } @Bean public MongoClient mongoClient(DynamicDataSource dynamicDataSource) { return dynamicDataSource; } @Bean public MongoTemplate mongoTemplate(DynamicDataSource dynamicDataSource) { return new MongoTemplate(dynamicDataSource); } } ``` 至此,就完成了 Spring Boot 整合 MongoDB 多数据源动态切换的配置。在需要切换数据源的地方,可以使用 `DataSourceContextHolder.setDataSource("db1")` 进行动态切换。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值