麻豆小视频在线观看_中文黄色一级片_久久久成人精品_成片免费观看视频大全_午夜精品久久久久久久99热浪潮_成人一区二区三区四区

首頁 > 開發 > Java > 正文

解決SpringBoot項目使用多線程處理任務時無法通過@Autowired注入bean問題

2024-07-14 08:42:13
字體:
來源:轉載
供稿:網友

最近在做一個“溫濕度控制”的項目,項目要求通過用戶設定的溫濕度數值和實時采集到的數值進行比對分析,因為數據的對比與分析是一個通過前端頁面控制的定時任務,經理要求在用戶開啟定時任務時,單獨開啟一個線程進行數據的對比分析,并將采集到的溫濕度數值存入數據庫中的歷史數據表,按照我們正常的邏輯應該是用戶在請求開啟定時任務時,前端頁面通過調用后端接口,創建一個新的線程來執行定時任務,然后在線程類中使用 @Autowired 注解注入保存歷史數據的service層,在線程類中調用service層保存歷史數據的方法實現溫濕度數據的保存,這時就出現了一個很尷尬的問題,在新開啟的線程中使用 @Autowired 注解無法注入需要的bean(即:保存歷史數據的service層),程序一直在報 NullPointerException 。

這是controller層,方法 startExperiment 和 stopExperiment 分別是開始定時任務和停止定時任務的方法,getData方法不屬于本次討論范圍,不用管

package com.backstage.controller;import com.alibaba.fastjson.JSONObject;import com.backstage.entity.JsonResponse;import com.backstage.entity.Threshold;import com.backstage.service.MainPageService;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.web.bind.annotation.RequestMapping;import org.springframework.web.bind.annotation.RestController;import javax.servlet.http.HttpServletRequest;/** * @ProjectName: * @Package: com.backstage.controller * @ClassName: MainPageController * @Description: 主頁面相關操作控制器 * @Author: wangzhilong * @CreateDate: 2018/8/29 9:49 * @Version: 1.0 */@RestController@RequestMapping("/main")public class MainPageController { @Autowired private MainPageService mainPageService; /**  * 開始實驗  *  * @param threshold  */ @RequestMapping("/startExperiment") public JsonResponse startExperiment(HttpServletRequest request, Threshold threshold) {  return mainPageService.startExperiment(request, threshold); } /**  * 停止實驗  */ @RequestMapping("/stopExperiment") public JsonResponse stopExperiment() {  return mainPageService.stopExperiment(); } /**  * 獲取實時數據  *  * @return  */ @RequestMapping("/getData") public JSONObject getData() {  return null; }}

 service 層接口代碼,沒什么好說的,直接上代碼:

package com.backstage.service;import com.alibaba.fastjson.JSONObject;import com.backstage.entity.JsonResponse;import com.backstage.entity.Threshold;import javax.servlet.http.HttpServletRequest;/** * @ProjectName:  * @Package: com.backstage.service * @ClassName: MainPageService * @Description: 主頁面相關操作業務層接口 * @Author: wangzhilong * @CreateDate: 2018/8/29 9:51 * @Version: 1.0 */public interface MainPageService { /**  * 開始實驗  *  * @param threshold  */ JsonResponse startExperiment(HttpServletRequest request, Threshold threshold); /**  * 停止實驗  */ JsonResponse stopExperiment(); /**  * 獲取實時數據  *  * @return  */ JSONObject getData();}

 

 service 層實現類代碼,關于springboot項目使用多線程進行業務處理不屬于本章節的討論范圍,如有需要,請留言,我會在看到留言后第一時間更新相關技術文章,由于這里刪除了一些與本章節無關的代碼,如果復制到開發工具內有報錯問題,麻煩大家提醒我一下,以便修改,非常感謝

package com.backstage.service.impl;import com.alibaba.fastjson.JSONObject;import com.backstage.entity.*;import com.backstage.monitor.TimingMonitoring;import com.backstage.service.*;import org.springframework.beans.factory.annotation.Autowired;import org.springframework.context.annotation.Bean;import org.springframework.scheduling.Trigger;import org.springframework.scheduling.TriggerContext;import org.springframework.scheduling.concurrent.ThreadPoolTaskScheduler;import org.springframework.scheduling.support.CronTrigger;import org.springframework.stereotype.Service;import javax.servlet.http.HttpServletRequest;import java.text.SimpleDateFormat;import java.util.Date;import java.util.List;import java.util.concurrent.ScheduledFuture;/** * @ProjectName:  * @Package: com.backstage.service.impl * @ClassName: MainPageServiceImpl * @Description: 主頁面相關操作業務層實現類 * @Author: wangzhilong * @CreateDate: 2018/8/29 9:51 * @Version: 1.0 */@Servicepublic class MainPageServiceImpl implements MainPageService { @Autowired private ThreadPoolTaskScheduler threadPoolTaskScheduler; private ScheduledFuture<?> future2; @Bean public ThreadPoolTaskScheduler threadPoolTaskScheduler() {  return new ThreadPoolTaskScheduler(); } /**  * 開始實驗  *  * @param threshold  */ @Override public JsonResponse startExperiment(HttpServletRequest request, Threshold threshold) {  TimingMonitoring timingMonitoring = new TimingMonitoring();  timingMonitoring.setThreshold(threshold, list, experiment.getId(), experimentData.getId());  future2 = threadPoolTaskScheduler.schedule(new TimingMonitoring(), new Trigger() {   @Override   public Date nextExecutionTime(TriggerContext triggerContext) {    //設置定時任務的執行時間為3秒鐘執行一次    return new CronTrigger("0/10 * * * * ?").nextExecutionTime(triggerContext);   }  });  return new JsonResponse(0,"開始實驗!"); } /**  * 停止實驗  */ @Override public JsonResponse stopExperiment() {  if (future2 != null) {   experimentService.upd(getTime());   future2.cancel(true);  }  return new JsonResponse(0,"結束實驗!"); } /**  * 獲取實時數據  *  * @return  */ @Override public JSONObject getData() {  return null; } protected String getTime() {  SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");  return format.format(new Date()); }}

重點,線程類代碼,大家注意看,我在代碼最開始使用了spring的 @Autowired 注解注入需要的service,可在調用service中的add方法時,程序報空指針異常,一直認為是add方法或者sql語句有問題,找了一上午,也沒發現任何問題,后來單獨調用這個add方法是可以正常插入數據的,唯獨在這個線程類中調用時報錯,感覺和線程有莫大的關系,百度一搜,還真找到了,原來,在線程中為了線程安全,是防注入的,沒辦法,要用到這個類啊。只能從bean工廠里拿個實例了,繼續往下看

package com.backstage.monitor;import com.backstage.entity.DetailedData;import com.backstage.entity.Threshold;import com.backstage.entity.ValveValue;import com.backstage.service.DetailedDataService;import java.text.SimpleDateFormat;import java.util.Date;import java.util.List;/** * @ProjectName: * @Package: com.backstage.monitor * @ClassName: TimingMonitoring * @Description: 定時監測溫(濕)度 數據 * @Author: wangzhilong * @CreateDate: 2018/8/29 10:11 * @Version: 1.0 */public class TimingMonitoring implements Runnable{ //歷史數據業務層接口 @Autowired public DetailedDataService detailedDataService; private Threshold threshold;   //閾值實體類 private List<ValveValue> settingData; //設定的溫濕度數據 private Integer id;      //實驗記錄id private Integer dataId;     //歷史數據主表id public void setThreshold(Threshold threshold, List<ValveValue> settingData, Integer id, Integer dataId) {  this.threshold = threshold;  this.settingData = settingData;  this.id = id;  this.dataId = dataId; } @Override public void run() {  //模擬從PLC獲取到的數據  String data = "001,50.5,002,37,003,45.6,004,40,005,55.2,006,58";  if (data == null || data.trim() == "") {   return; //若獲取到的數據為空,則直接停止該方法的執行  }  double temperature = 0.0; //溫度  double humidity = 0.0;  //濕度  Integer type = null;    //數據類型,1是溫度,2是濕度  //解析數據,并將數據保存到歷史數據數據庫  String[] str = data.split(",");  SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss:SS");  for (int i = 0; i < str.length; i++) {   if (i == 1 || i == 5 || i == 9) { //溫度    type = 1;    temperature += Double.parseDouble(str[i]);    //System.out.println("溫度" + i + " -》 " + str[i-1] + ":" + str[i]);    detailedDataService.add(new DetailedData(null, type, Double.parseDouble(str[i]), format.format(new Date()), str[i - 1], dataId));   }   if (i == 3 || i == 7 || i == 11) { //濕度    type = 2;    humidity += Double.parseDouble(str[i]);    //System.out.println("濕度" + i + " -》 " + str[i-1] + ":" + str[i]);    detailedDataService.add(new DetailedData(null, type, Double.parseDouble(str[i]), format.format(new Date()), str[i - 1], dataId));   }  } } /**  * 獲取當前時間,精確到毫秒  * @return  */ protected String getTime() {  SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss:SS");  return format.format(new Date()); }}

獲取bean對象的工具類,既然程序無法通過注解拿到需要的bean,那就只好自己寫個工具類來獲取嘍,下面是工具類代碼

package com.backstage.config;import org.springframework.beans.BeansException;import org.springframework.context.ApplicationContext;import org.springframework.context.ApplicationContextAware;import org.springframework.stereotype.Component;/** * @ProjectName: * @Package: com.backstage.config * @ClassName: ApplicationContextProvider * @Description: 獲取bean對象的工具類 * @Author: wangzhilong * @CreateDate: 2018/8/31 13:26 * @Version: 1.0 *//** * Author:ZhuShangJin * Date:2018/7/3 */@Componentpublic class ApplicationContextProvider implements ApplicationContextAware { /**  * 上下文對象實例  */ private static ApplicationContext applicationContext; @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {  this.applicationContext = applicationContext; } /**  * 獲取applicationContext  *  * @return  */ public static ApplicationContext getApplicationContext() {  return applicationContext; } /**  * 通過name獲取 Bean.  *  * @param name  * @return  */ public static Object getBean(String name) {  return getApplicationContext().getBean(name); } /**  * 通過class獲取Bean.  *  * @param clazz  * @param <T>  * @return  */ public static <T> T getBean(Class<T> clazz) {  return getApplicationContext().getBean(clazz); } /**  * 通過name,以及Clazz返回指定的Bean  *  * @param name  * @param clazz  * @param <T>  * @return  */ public static <T> T getBean(String name, Class<T> clazz) {  return getApplicationContext().getBean(name, clazz); }}

這樣呢,就可以在線程類中寫一個無參的構造方法,在構造方法中,通過調用工具類中的 getBean() 方法就可以拿到實例了,程序在調用這個線程類時,會自動調用其無參的構造方法,在構造方法中我們將需要的bean對象注入,然后就可以正常使用了,下邊是線程類修改后的代碼,由于別的地方沒有改動,所以這里只給大家改動的代碼,省得大家看到一大堆代碼頭疼。

public TimingMonitoring() {  //new的時候注入需要的bean  this.detailedDataService = ApplicationContextProvider.getBean(DetailedDataService.class); }

總結

以上所述是小編給大家介紹的SpringBoot項目使用多線程處理任務時無法通過@Autowired注入bean 問題,希望對大家有所幫助,如果大家有任何疑問歡迎給我留言,小編會及時回復大家的!


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
主站蜘蛛池模板: 亚洲国产超高清a毛毛片 | 国产精品视频免费网站 | 日美黄色片 | 成人精品一区二区 | 国产一级做a | 国产精品久久久免费 | 久久精品视频69 | 久久精品视频16 | 久久成人亚洲 | 免费观看一区 | 91精品国产成人 | 高清做爰免费无遮网站挡 | 羞羞视频一区二区 | 在线播放av片 | av在线免费播放网站 | 亚洲天堂岛国片 | 中国hdxxxx护士爽在线观看 | 少妇一级淫片免费放4p | 国产一级毛片国语版 | 欧美成人高清视频 | 羞羞视频免费网站含羞草 | 日本黄色免费播放 | 精品无码一区在线观看 | 午夜国产精品成人 | 黄色大片在线免费观看 | 国产成年人小视频 | 在线播放的av网站 | 国产午夜精品久久久久婷 | 日韩视频在线观看免费 | www.精品久久 | 日日狠狠久久偷偷四色综合免费 | 国产一区二区免费在线观看视频 | 色婷婷a v | 91看片淫黄大片欧美看国产片 | 国产69精品久久久久久野外 | 午夜色片 | 18视频在线观看娇喘 | 亚洲国产高清一区 | 亚洲视频黄| 国产一级爱c视频 | 国产在线精品一区二区夜色 |