1、概述
一般來說,springboot工程環(huán)境配置放在properties文件中,啟動的時候?qū)⒐こ讨械膒roperties/yaml文件的配置項(xiàng)加載到內(nèi)存中。但這種方式改配置項(xiàng)的時候,需要重新編譯部署,考慮到這種因素,今天介紹將配置項(xiàng)存到數(shù)據(jù)庫表中,在工程啟動時把配置項(xiàng)加載到內(nèi)存中。
springboot提供了兩個接口: CommandLineRunner 和 ApplicationRunner 。實(shí)現(xiàn)其中接口,就可以在工程啟動時將數(shù)據(jù)庫中的數(shù)據(jù)加載到內(nèi)存。使用的場景有:加載配置項(xiàng)到內(nèi)存中;啟動時將字典或白名單數(shù)據(jù)加載到內(nèi)存(或緩存到Redis中)。
ApplicationRunner 與CommandLineRunner工作方式相同,唯一的區(qū)別在于兩種方法入?yún)⒎绞讲煌瑢?shí)現(xiàn)這兩個接口就可以讓應(yīng)用程序代碼在啟動完成后,接收流量前被調(diào)用。如果實(shí)現(xiàn)了多次,則必須實(shí)現(xiàn)org.springframework.core.Ordered或者使用org.springframework.core.annotation.Order注解來定義他們之間的順序,@Order(1)數(shù)字越小優(yōu)先級越高
不使用注解@Order執(zhí)行順序:
靜態(tài)代碼塊?> 構(gòu)造代碼塊?> 構(gòu)造方法(Constructor) > @Autowired > @PostConstruct? >?ApplicationRunner >?CommandLineRunner
推薦使用: ApplicationRunner 或者 CommandLineRunner方式
springbean初始化流程
st=>start: 開始
op1=>operation: Spring容器加載
op2=>operation: 調(diào)用構(gòu)造函數(shù)
op3=>operation: @PostConstruct方法調(diào)用
op4=>operation: init()調(diào)用
op5=>operation: 其他代碼
op6=>operation: destroy()調(diào)用
op7=>operation: @PreDestroy方法調(diào)用
e=>end: 結(jié)束
st->op1->op2->op3->op4->op5->op6->op7->e
靜態(tài)代碼塊:僅在類加載時執(zhí)行一次。
構(gòu)造代碼塊:創(chuàng)建一次對象,執(zhí)行一次
靜態(tài)方法里不能調(diào)用類的實(shí)例方法,但可以調(diào)用構(gòu)造方法(因?yàn)椴恍枰獙?shí)例)
2、加載方式
2.1、ApplicationRunner接口
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) // 初始化加載優(yōu)先級,數(shù)字越小優(yōu)先級越高
public class InitData implements ApplicationRunner {
private static Map<String,Dictionary> dataMap = new HashMap<>();
@Autowired
private DictionaryDao dictionaryDao;
@Override
public void run(ApplicationArguments args) throws Exception {
// 字典信息放到map中
List<Dictionary> dicList = dictionaryDao.findAllDictionary();
for(Dictionary dic : dicList){
dataMap.put(dic.getKey,dic);
}
}
// 獲取字典信息的具體方法
public static String getDicDataByKey(String key){
Dictionary dictionary = dataMap.get(dataMap);
return dictionary ;
}
}
?2.2、CommandLineRunner接口
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) // 初始化加載優(yōu)先級,數(shù)字越小優(yōu)先級越高
public class InitDicData implements CommandLineRunner {
private static Map<String,Dictionary> dataMap = new HashMap<>();
@Autowired
private DictionaryDao dictionaryDao;
@Override
public void run(String... args) throws Exception {
// 字典信息放到map中
List<Dictionary> dicList = dictionaryDao.findAllDictionary();
for(Dictionary dic : dicList){
dataMap.put(dic.getKey,dic);
}
}
// 獲取字典信息的具體方法
public static String getDicDataByKey(String key){
Dictionary dictionary = dataMap.get(dataMap);
return dictionary ;
}
}
2.3、@PostConstruct注解
很多人濫用該注解,并且認(rèn)為該注解是基于Spring的。其實(shí)不然,@PostConstruct和@PreDestroy是在Java EE 5引入并且基于Servlet規(guī)范, 位于javax.annotation包下,Java最初的設(shè)計者認(rèn)為,這些功能并不是Java核心API,因此就放到了擴(kuò)展包中。該注解執(zhí)行的優(yōu)先級很高,在ApplicationRunner接口之前執(zhí)行。?
@PostConstruct注解修飾的方法會在服務(wù)器加載Servlet的時候運(yùn)行,并且只會被服務(wù)器調(diào)用一次,類似于Servlet的init()方法,被@PostConstruct注解修飾的方法會在構(gòu)造函數(shù)之后,init()方法執(zhí)行之前執(zhí)行文章來源:http://www.zghlxwxcb.cn/news/detail-798760.html
被@PreDestroy注解修飾的方法會在服務(wù)器卸載Servlet的時候運(yùn)行,并且只會被服務(wù)器調(diào)用一次,類似于Servlet的destroy()方法,被@PreDestroy注解修飾的方法會在destroy()方法之后、在Servlet被徹底卸載之前執(zhí)行文章來源地址http://www.zghlxwxcb.cn/news/detail-798760.html
@Component
@Slf4j
public class CacheDicUtils {
private static Map<String, Dictionary> dataMap = new HashMap<>();
@Autowired
private DictionaryDao dictionaryDao;
@PostConstruct
public void init() {
List<Dictionary> list = dictionaryDao.findAllDictionary();
for (Dictionary dictionary : list) {
dataMap.put(dictionary.getName(), dictionary);
}
}
@PreDestroy
public void destroy() {
log.info("系統(tǒng)啟動成功,dataMap加載完成!");
}
public static Dictionary getDictDataByName(String name) {
Dictionary dictionary = dataMap.get(name);
return sysDictionary ;
}
}
2.4、static靜態(tài)代碼塊
public class DicMap {
@Autowired
private static DictionaryDao dictionaryDao;
private static HashMap<String,Dictionary> dataMap = new HashMap<>();
static {
// 在這里我們不能使用注入進(jìn)來的dictionaryDao,因?yàn)樗藭r還沒有被創(chuàng)建出來,所以我們要在系統(tǒng)
// 運(yùn)行的時候獲取項(xiàng)目的上下文手動去獲取這個bean
dictionaryDao = SpringUtil.getBean(DictionaryDao.class);
queryDic();
}
// 把字典信息放到map中
public static void queryDic(){
List<Dictionary> dicList = dictionaryDao.findAllDictionary();
for(Dictionary dic : dicList){
dataMap.put(dic.getKey,dic);
}
}
// 獲取字典信息的具體方法
public static String getDicDataByKey(String key){
Dictionary dictionary = dataMap.get(dataMap);
return dictionary ;
}
}
到了這里,關(guān)于springboot啟動加載數(shù)據(jù)庫數(shù)據(jù)到內(nèi)存的文章就介紹完了。如果您還想了解更多內(nèi)容,請在右上角搜索TOY模板網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章,希望大家以后多多支持TOY模板網(wǎng)!