Spring Boot 动态指定数据库,实现不同用户不同数据库等场景

当在 Spring Boot 应用程序中使用Spring Data JPA 进行数据库操作时,配置Schema名称是一种常见的做法。然而,在某些情况下,模式名称需要是动态的,可能会在应用程序运行时发生变化。比如:需要做数据隔离的SaaS应用。

所以,这篇博文将帮助您解决了在 Spring Boot 应用程序中如何设置动态 Schema。

#问题场景

假设,您的应用程序是一个SaaS软件,需要为多个租户提供服务,每个租户都需要一个单独的数据库架构。

在这种情况下,在应用程序属性中对Shema名称进行硬编码是不太可能的,这样有一个用户新增,就要去写代码更新。

所以,为了应对这一挑战,我们将探索一种允许在运行时动态配置模式名称的解决方案。

#代码案例

让我们创建一个 Spring Boot 项目 首先设置一个具有必要依赖项的新 Spring Boot 项目。在项目配置中包括 Spring Web、Spring Data JPA 和关于数据库的依赖项。

定义Spring Data JPA的实体类,例如:

 

import javax.persistence.Entity;

import javax.persistence.Id;

import javax.persistence.Table;

 

@Entity

@Table(name = "product")

public class Product {

    @Id

    private Long id;

    private String name;

    private double price;

 

}

创建数据访问接口,以便您的实体提供 CRUD 操作:

 

import org.springframework.data.jpa.repository.JpaRepository;

import org.springframework.stereotype.Repository;

 

@Repository

public interface ProductRepository extends JpaRepository<Product, Long> {

 

}

创建一个用来处理业务逻辑,包括与数据库交互的方法:

 

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

import org.springframework.stereotype.Service;

 

import java.util.List;

 

@Service

public class ProductService {

    private final ProductRepository productRepository;

 

    @Autowired

    public ProductService(ProductRepository productRepository) {

        this.productRepository = productRepository;

    }

 

    public List<Product> getAllProducts() {

        return productRepository.findAll();

    }

 

}

实现API接口:

 

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

import org.springframework.web.bind.annotation.GetMapping;

import org.springframework.web.bind.annotation.RequestMapping;

import org.springframework.web.bind.annotation.RestController;

 

import java.util.List;

 

@RestController

@RequestMapping("/api/products")

public class ProductController {

    private final ProductService productService;

 

    @Autowired

    public ProductController(ProductService productService) {

        this.productService = productService;

    }

 

    @GetMapping

    public List<Product> getAllProducts() {

        return productService.getAllProducts();

    }

 

}

重点:配置动态Schema

 

import org.springframework.beans.factory.annotation.Value;

import org.springframework.boot.jdbc.DataSourceBuilder;

import org.springframework.context.annotation.Bean;

import org.springframework.context.annotation.Configuration;

 

import javax.sql.DataSource;

 

@Configuration

public class DynamicSchemaConfig {

 

    @Value("${custom.schema.name}")

    private String customSchemaName;

 

    @Bean

    public DataSource dataSource() {

        String dataSourceUrl = "jdbc:mysql://localhost:3306/" + customSchemaName;

        return DataSourceBuilder.create().url(dataSourceUrl).build();

    }

}

重新打包该Spring Boot应用,然后当我们要为不同用户使用完全隔离的数据库、完全隔离的应用的时候,只需要通过下面的启动命令,就能轻松实现了:

 

java -jar -Dcustom.schema.name=my_dynamic_schema your-application.jar

这里,通过启动命令中的custom.schema.name参数,就能去指定不同的数据库Schema,而应用程序端都是同一套代码,由于启动了新的Spring Boot应用,所以应用端进程也是完全隔离的。这种方法,对于使用Spring Boot构建需要一定资源隔离SaaS软件来说,是个不错的实现方案。

  • 7
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Spring Boot中创建定时任务可以使用注解 `@EnableScheduling` 开启定时任务的支持。首先,在启动类上添加 `@EnableScheduling` 注解。接下来,创建一个定时任务的类,并添加 `@Component` 注解,使其成为Spring管理的Bean。 在定时任务类中,可以使用`@Scheduled`注解来标注具体的定时方法,以指定任务的执行时间。`@Scheduled`注解支持多种时间表达方式,如固定延迟时间、固定间隔时间、Cron表达式等。 在配合数据库动态执行定时任务的场景中,可以在任务方法中读取数据库中的配置信息,来动态设定定时任务的执行时间。具体实现方式如下: 1. 首先,在数据库中创建一个定时任务配置表,包含任务名称、任务执行时间等字段。 2. 在定时任务类中注入数据源,并使用JdbcTemplate或者MyBatis等方式来访问数据库。 3. 创建一个方法,通过查询数据库获取定时任务的执行时间,并将执行时间作为参数传递给`@Scheduled`注解。 4. 使用`@Scheduled`注解标注这个方法,使其成为定时执行的任务。 以下是一个示例代码: ```java @Component public class MyScheduledTask { @Autowired private JdbcTemplate jdbcTemplate; @Scheduled(fixedDelay = 5000) // 每隔5秒执行一次 public void executeTask() { // 从数据库读取任务执行时间 String scheduleTime = jdbcTemplate.queryForObject("SELECT schedule_time FROM task_config WHERE task_name = 'myTask'", String.class); // 将任务执行时间格式化为Cron表达式 String cronExpression = convertToCronExpression(scheduleTime); // 执行任务 System.out.println("执行定时任务..."); // 更新数据库中的任务执行状态等相关信息 jdbcTemplate.update("UPDATE task_config SET last_execute_time = ?, status = 'completed' WHERE task_name = 'myTask'", new Date()); } private String convertToCronExpression(String scheduleTime) { // 将任务执行时间格式转换为Cron表达式的逻辑 // ... } } ``` 以上代码中,定时任务类`MyScheduledTask`中的`executeTask`方法使用`@Scheduled`注解标注,将定时任务的执行时间动态设定为从数据库中读取的值。执行任务的代码可以根据具体的业务需求进行编写。 需要注意的是,在使用数据库动态设定定时任务的执行时间时,需要在适当的时候更新数据库中相关的任务信息,以便下次执行任务时能够获取到最新的执行时间。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值