java线程池方式执行run任务及在线程run方法中注入service

背景:接口1内需要调用接口2,接口2功能是执行一个长时间任务,执行完成后才会返回结果(比如执行5min后获取到执行结果),但是接口1内调接口2时不能等待5min后才返回结果,接口1需要返回自己的结果

方式:以线程池方式来执行任务,接口1触发接口2后不等待,直接返回自己的内容或做其他操作,将接口2的任务扔进线程池中任其执行,执行完了后可以接着做其他操作

一、线程池方式执行run任务

例子:

2个数据库表:report(存储报告连接)、task(任务表,包含任务执行的状态status和报告id)

1、创建业务接口controller

package com.controller;

import com.alibaba.fastjson.JSONObject;
import com.thtead.Worker;
import com.utils.ResultJson;
import com.utils.ResultGenerator;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.*;

import java.util.concurrent.*;

@RestController
@RequestMapping("api/v1")
public class TestController {

    //创建线程池
    public static ThreadPoolExecutor executorService = new ThreadPoolExecutor(5, 5, 60L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<>(1));


    /**
     * 线程池执行任务
     * @return
     * @throws Exception
     */
    @PostMapping("/run")
    @ResponseBody
    public JSONObject post() throws Exception {

        //做自己的处理,比如先生成一条task记录,写入task表,将report id置空,status值为running
        JSONObject json =new JSONObject();
        json.put("taskid","111");
        //调用worker在线程池中执行任务
        executorService.submit(new Worker());   

        //不等待,做自己的处理,比如该接口返回task id
        return json;
    }


}

2、接口2:真正执行任务

package com.controller;

import com.alibaba.fastjson.JSONObject;
import org.springframework.web.bind.annotation.*;

@RestController
@RequestMapping("api/v1")
public class TestController2 {


    @RequestMapping(value = "/Report", method = RequestMethod.POST)
    @ResponseBody
    public JSONObject jacocoReport() throws Exception {

        Thread.sleep(6000);   //模拟任务执行,即需要长时间执行
        //模拟任务执行完后的返回内容
        JSONObject json =new JSONObject();
        json.put("key2","value2");
        json.put("key3","value3");
        JSONObject json2 =new JSONObject();
        json2.put("reporturl","http://111.231231.32321/3213123/312/31");
        json2.put("key2",json);
        return json2;

    }
}

3、创建worker,复写run

package com.thtead;

import com.alibaba.fastjson.JSONObject;
import com.controller.TestController2;


public class Worker implements Runnable {
    //用于传参    
    public Worker(String page){
        this.page1 = page;
    }
    String page1;

    @Override
    public void run(){
        try {
            TestController2 controller2 = new TestController2();
            JSONObject json = controller2.jacocoReport();   //执行任务
            System.out.println(json.get("reporturl"));
            System.out.println(page1);
            //这里可以继续执行其他操作:等待任务执行完获取到结果后,将结果接入report表,生成report id
            //再将report id写入task表,同时更改task表中的status为finish

        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

二、在线程run方法中注入service

 采坑: 在线程的run方法中使用@Autowired注解获得Service层实例,调用service层方法是报空指针:

 

 原因:在线程的run方法中使用注解@autoware注入的bean,会报空指针异常,原因是因为线程中为了线程安全,防注入。

解决:在bean工厂中拿实例

1、新增获取bean实例的工具类组件

package com.utils;

import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;

/**
 * 在线程的run方法中使用注解@autoware注入的bean,会报空指针异常,原因是因为线程中为了线程安全,防注入。
 * 获取bean实例的工具类组件
 */
@Component
public class GetBeanUtil implements ApplicationContextAware {
    //Spring应用上下文环境
    private static ApplicationContext applicationContext;
    /**
     * 实现ApplicationContextAware接口的回调方法,设置上下文环境
     */
    public void setApplicationContext(ApplicationContext context) {
        GetBeanUtil.applicationContext = context;
    }
    /**
     * 获取对象 这里重写了bean方法,起主要作用
     */
    public static Object getBean(String name) {
        return applicationContext.getBean(name);
    }

    public static ApplicationContext getApplicationContext() {
        return applicationContext;
    }
}

2、在run方法中使用工具类拿到Service层实例:

package com.thtead;

import com.alibaba.fastjson.JSONObject;
import com.controller.TestController2;
import com.service.ApiService;
import com.utils.GetBeanUtils;


public class Worker implements Runnable {

    //不要使用 @Autowired
    private ApiService apiService;

    public Worker(String page){
        this.page1 = page;
    }
    String page1;
    @Override
    public void run(){
        this.apiService = GetBeanUtils.getApplicationContext().getBean(ApiService.class);  //可以使用apiService里的接口
        try {
            TestController2 controller2 = new TestController2();
            JSONObject json = controller2.jacocoReport();
            System.out.println(json.get("reporturl"));
            System.out.println(page1);
            //这里可以继续执行其他操作:等待任务执行完获取到结果后,将结果接入report表,生成report id
            //再将report id写入task表,同时更改task表中的status为finish
            apiService.selectApiInfo("213123");
//             http  run -》reporturl
//                写库
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

}

  • 3
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 4
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值