一般来说,springboot工程环境配置放在properties文件中,启动的时候将工程中的properties/yaml文件的配置项加载到内存中。但这种方式改配置项的时候,需要重新编译部署,考虑到这种因素,今天介绍将配置项存到数据库表中,在工程启动时把配置项加载到内存中。
springboot提供了两个接口: CommandLineRunner 和 ApplicationRunner 。实现其中接口,就可以在工程启动时将数据库中的数据加载到内存。使用的场景有:加载配置项到内存中;启动时将字典或白名单数据加载到内存(或缓存到Redis中)。
springbean初始化流程
st=>start: 开始
op1=>operation: Spring容器加载
op2=>operation: 调用构造函数
op3=>operation: @PostConstruct方法调用
op4=>operation: init()调用
op5=>operation: 其他代码
op6=>operation: destroy()调用
op7=>operation: @PreDestroy方法调用
e=>end: 结束
st->op1->op2->op3->op4->op5->op6->op7->e
@PostConstruct注解修饰的方法会在服务器加载Servlet的时候运行,并且只会被服务器调用一次,类似于Servlet的init()方法,被@PostConstruct注解修饰的方法会在构造函数之后,init()方法执行之前执行
被@PreDestroy注解修饰的方法会在服务器卸载Servlet的时候运行,并且只会被服务器调用一次,类似于Servlet的destroy()方法,被@PreDestroy注解修饰的方法会在destroy()方法之后、在Servlet被彻底卸载之前执行
@Component
@Slf4j
public class CacheDicUtils {
private static Map<String, SysDictionary> dataMap = new HashMap<>();
@Autowired
private SysDictionaryMapper sysDictionaryMapper;
@PostConstruct
public void init() {
QueryWrapper<SysDictionary> wrapper = new QueryWrapper<>();
List<SysDictionary> list = sysDictionaryMapper.selectList(wrapper);
for (SysDictionary sysDictionary : list) {
dataMap.put(sysDictionary.getName(), sysDictionary);
}
}
public static SysDictionary getDictDataByName(String name) {
SysDictionary sysDictionary = dataMap.get(name);
return sysDictionary ;
}
}
package com.example.demo.config;
import com.example.demo.service.ICodeService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.CommandLineRunner;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
@Component
@Order(1) // 初始化加载优先级,数字越小优先级越高
public class InitData2 implements CommandLineRunner {
public static Map<Integer, String> codeMap = new HashMap<Integer, String>();
@Autowired
private ICodeService codeService;
@Override
public void run(String... args) throws Exception {
System.out.println("示例2:加载codeMap中......");
// 查询数据库数据
List<String> codeList = codeService.listAll();
for (int i = 0; i < codeList.size(); i++) {
codeMap.put(i, codeList.get(i));
}
}
}
package com.example.demo.config;
import com.example.demo.service.ICodeService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.ApplicationArguments;
import org.springframework.boot.ApplicationRunner;
import org.springframework.core.annotation.Order;
import org.springframework.stereotype.Component;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
@Component
@Order(1) // 初始化加载优先级,数字越小优先级越高
public class InitData3 implements ApplicationRunner {
public static Map<Integer, String> codeMap = new HashMap<Integer, String>();
@Autowired
private ICodeService codeService;
@Override
public void run(ApplicationArguments args) throws Exception {
System.out.println("示例3:加载codeMap中......");
// 查询数据库数据
List<String> codeList = codeService.listAll();
for (int i = 0; i < codeList.size(); i++) {
codeMap.put(i, codeList.get(i));
}
}
}
public class DicMap {
@Autowired
private static CustomerDao customerDao;
private static HashMap<String,Dictionary> dataMap = new HashMap<>();
static {
// 在这里我们不能使用注入进来的customerDao,因为它此时还没有被创建出来,所以我们要在系统
// 运行的时候获取项目的上下文手动去获取这个bean
customerDao = SpringUtil.getBean(CustomerDao.class);
queryDic();
}
// 把字典信息放到map中
public static void queryDic(){
List<Dictionary> dicList = customerDao.findAllDictionary();
for(Dictionary dic : dicList){
dataMap.put(dic.getKey,dic);
}
}
// 获取字典信息的具体方法
public static String getDicDataByKey(String key){
Dictionary dictionary = dataMap.get(dataMap);
return dictionary ;
}
}