網(wǎng)站首頁 編程語言 正文
一招教你如何減少本地調(diào)試tomcat重啟次數(shù)
當(dāng)我們進(jìn)行本地調(diào)試的時候,代碼做了少量改動,卻要重啟tomcat。如果項目比較小還行,如果項目比較大這個時候重啟tomcat的時間就比較長。下面我說的方法將會讓你減少tomcat不必要的重啟次數(shù)。
這次引入的技術(shù)為Groovy。
在groovy中書寫的代碼無需重啟tomcat,修改之后需需要重新從入口進(jìn)入就行了
什么是Gooovy
Apache Groovy是一種功能強(qiáng)大、可選的類型和動態(tài)語言,具有靜態(tài)鍵入和靜態(tài)編譯功能,適用于Java平臺,旨在通過簡潔、熟悉和易于學(xué)習(xí)的語法提高開發(fā)人員的工作效率。它與任何Java程序順利集成,并立即為您的應(yīng)用程序提供強(qiáng)大的功能,包括腳本功能、特定域語言創(chuàng)作、運(yùn)行時和編譯時元編程以及功能編程。和Java兼容性強(qiáng),可以無縫銜接Java代碼,可以調(diào)用Java所有的庫。
多得不說,直接上代碼
pom依賴
<dependency> <groupId>org.codehaus.groovy</groupId> <artifactId>groovy-jsr223</artifactId> <version>3.0.6</version> </dependency>
Controller
@Controller @Slf4j public class ScriptAction { @Autowired private GroovyEval groovyEval; @RequestMapping(value = "/script/test") //入?yún)ⅲ篻roovy腳本存放絕對路徑、需要傳遞的參數(shù) public Object scriptTest( @Param(value = "path", required = true) String path, @Json("@requestBody") @RequestBody Map<String,Object> paramMap ) { try { BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(new FileInputStream(path), StandardCharsets.UTF_8)); String date; StringBuilder stringBuilder = new StringBuilder(); while((date = bufferedReader.readLine()) != null){ stringBuilder.append(date).append("\n"); } bufferedReader.close(); //執(zhí)行腳本獲得結(jié)果,約定執(zhí)行的腳本方法名字為solution return groovyEval.evalScript(bufferedReader.toString() , "solution" , new Object[]{paramMap}); } catch (FileNotFoundException e) { e.printStackTrace(); } catch (IOException e) { e.printStackTrace(); } return null; } }
Service
import com.google.gson.Gson; import groovy.lang.GroovyClassLoader; import groovy.lang.GroovyObject; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.BeansException; import org.springframework.context.ApplicationContext; import org.springframework.context.ApplicationContextAware; import org.springframework.stereotype.Component; import org.springframework.util.CollectionUtils; import java.util.Map; import java.util.concurrent.TimeUnit; @Slf4j @Component public class GroovyEval implements ApplicationContextAware { private static GroovyEval groovyEval; private ApplicationContext applicationContext; public static <T> T getBean(Class<T> cls){ return groovyEval.applicationContext.getBean(cls); } public Object evalScript(String script, String methodName, Object[] args){ Object scriptObj = this.getScript(script); try { //腳本執(zhí)行入口 //返回的數(shù)據(jù)類型在groovy腳本中自己定義即可,我這里返回的是map Map<String, Object> resultMap = (Map<String, Object>)((GroovyObject)scriptObj).invokeMethod(methodName, args); if (CollectionUtils.isEmpty(resultMap)){ return null; } return resultMap.get("data"); } catch (Throwable e) { log.error("script eval error !" , e); } return null; } private Object getScript(String script){ //注意?。。”镜卣{(diào)試可以不需要加入緩存機(jī)制,生產(chǎn)環(huán)境需要加入緩存 //加載腳本,每執(zhí)行一次new一個GroovyCodeSource Class<?> cls = new GroovyClassLoader().parseClass(script); GroovyObject groovyObject = null; try { log.info("load script!"); groovyObject = (GroovyObject)cls.newInstance(); } catch (IllegalAccessException | InstantiationException e) { log.error("load script error ! script : {}" , script , e); } return groovyObject; } @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { //靜態(tài)化bean this.applicationContext = applicationContext; groovyEval = this; } }
Groovy腳本
TestGroovy.groovy
class TestGroovy { def Map<String,Object> solution(Map<String,Object> paramMap){ Map<String,Object> resultMap = [:]; /** 獲取上層傳入的參數(shù) */ Object shopCodes = paramMap.get("param"); //業(yè)務(wù)邏輯處理。。。。。。 resultMap.put("data", "resultData"); return resultMap; } }
生產(chǎn)環(huán)境使用
因為groovy每執(zhí)行一次腳本,都會生成一個腳本的class對象,這個class對象的名字由 “script” + System.currentTimeMillis() +
Math.abs(text.hashCode())組成,因此應(yīng)用到生產(chǎn)環(huán)境需要加入緩存。推薦使用高性能緩存:Caffeine,
官方介紹Caffeine是基于JDK8的高性能本地緩存庫,提供了幾乎完美的命中率。它有點類似JDK中的ConcurrentMap,實際上,Caffeine中的LocalCache接口就是實現(xiàn)了JDK中的ConcurrentMap接口,但兩者并不完全一樣。最根本的區(qū)別就是,ConcurrentMap保存所有添加的元素,除非顯示刪除之(比如調(diào)用remove方法)。而本地緩存一般會配置自動剔除策略,為了保護(hù)應(yīng)用程序,限制內(nèi)存占用情況,防止內(nèi)存溢出。
有興趣的可以自己去搜索一下,我感覺蠻好用的
@Component public class GroovyEval implements ApplicationContextAware { private static final Logger LOGGER = LoggerFactory.getLogger(GroovyEval.class); private static final Object source = new Object(); private static GroovyEval groovyEval; private ApplicationContext applicationContext; @Autowired private AlarmThresholdSettingsItemService alarmThresholdSettingsItemService; public static <T> T getBean(Class<T> cls){ return groovyEval.applicationContext.getBean(cls); } private static final Cache<Object, Object> caffeine = Caffeine .newBuilder() .maximumSize(30000) //三天不用直接 gc .expireAfterAccess(72 , TimeUnit.HOURS) .build(); public Map lookUp(){ return caffeine.asMap(); } public Object evalScript(String script,String methodName,Object[] args) { Object scriptObj = this.getScript(script); if(scriptObj != null){ try{ //統(tǒng)一返回 Map<String,Object> { "data" : object } Map<String, Object> resultMap = (Map<String, Object>) ((GroovyObject) scriptObj).invokeMethod(methodName, args); if(CollectionUtils.isEmpty(resultMap)){ return null; } return resultMap.get("data"); }catch (Throwable e){ LOGGER.error("script eval error !" , e); } } return null; } //腳本加入緩存 private Object getScript(String script){ //唯一標(biāo)記 String cacheKey = DigestUtils.md5Hex(script); return caffeine.get(cacheKey, new Function<Object, Object>() { @Override public Object apply(Object key) { //避免變動導(dǎo)致并發(fā)問題 synchronized (source){ Class<?> cls = new GroovyClassLoader().parseClass(script); GroovyObject gObj = null; try { LOGGER.info("load script !"); gObj = (GroovyObject) cls.newInstance(); } catch (InstantiationException | IllegalAccessException e) { LOGGER.error("load script error ! script : {}" , script , e); } return gObj; } } }); } @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { //靜態(tài)化 Bean this.applicationContext = applicationContext; groovyEval = this; } }
總結(jié)
原文鏈接:https://blog.csdn.net/asd1358355022/article/details/122667701
相關(guān)推薦
- 2022-05-07 Python中l(wèi)ist列表的賦值方法及遇到問題處理_python
- 2022-04-28 C++Stack棧類模版實例詳解_C 語言
- 2023-01-29 Python安裝Talib庫的詳細(xì)圖文教程_python
- 2022-10-07 Pandas數(shù)據(jù)分析固定時間點和時間差_python
- 2023-03-18 C#調(diào)用dll報錯:無法加載dll,找不到指定模塊的解決_C#教程
- 2022-06-13 詳解Python如何利用Pandas與NumPy進(jìn)行數(shù)據(jù)清洗_python
- 2022-04-09 Spring事務(wù)管理之開啟聲明式事務(wù)
- 2023-05-22 pycharm中執(zhí)行.sh文件的方法步驟_python
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡單動態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支