7、spring boot + Maven + Restful 模拟异步处理Rest服务 提高服务器吞吐量

同步异步处理对比
7、spring boot + Maven + Restful 模拟异步处理Rest服务 提高服务器吞吐量_第1张图片

  1. 使用Callable进行异步处理,副线程写在主线程里面的 ,符合企业级开发的一般应用场景!AsyncController.java
package com.imooc.web.asnyc;

import java.util.concurrent.Callable;

import org.apache.commons.lang.RandomStringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.context.request.async.DeferredResult;

@RestController
public class AsyncController {
	private Logger logger = LoggerFactory.getLogger(getClass());
	@RequestMapping("/order")
	public Callable order() throws Exception {
	logger.info("主线程开启");
		Callable result = new Callable() {
			@Override
			public String call() throws Exception {
				logger.info("副线程开启");
				Thread.sleep(1000);
				logger.info("副线程关闭");
				return "success";
			}
		};
		logger.info("主线程关闭");
		return result;
	}
}

运行结果:
在这里插入图片描述

  1. 使用DefferedResult异步处理Rest服务,符合企业级开发的复杂应用场景!
    7、spring boot + Maven + Restful 模拟异步处理Rest服务 提高服务器吞吐量_第2张图片
    在这里插入图片描述

MockQueue.java 模拟消息队列对象

package com.imooc.web.asnyc;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;

/**
 * 模拟消息队列的对象
 * @author caijiajun
 * @date   2018年9月13日
 */
@Component
public class MockQueue {
	
	//下单的消息
	private String placeOrder;
	//订单完成的消息
	private String completeOrder;
	
	private Logger logger = LoggerFactory.getLogger(getClass());
	
	public String getPlaceOrder() {
		return placeOrder;
	}

	public void setPlaceOrder(String placeOrder) throws Exception {
		new Thread(() -> {
			logger.info("接到下单请求," + placeOrder);
			try {
				Thread.sleep(1000);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
			//服务器处理一秒后订单完成(相当于应用二)
			this.completeOrder = placeOrder;
			logger.info("下单请求处理完毕," +  placeOrder);
		}).start();
	}

	public String getCompleteOrder() {
		return completeOrder;
	}

	public void setCompleteOrder(String completeOrder) {
		this.completeOrder = completeOrder;
	}
	
	
}

DeferredResultHolder.java 两个线程之前传递对象

package com.imooc.web.asnyc;

import java.util.HashMap;
import java.util.Map;

import org.springframework.stereotype.Component;
import org.springframework.web.context.request.async.DeferredResult;

/**
 * Http请求和响应两个线程之间传递对象
 * @author caijiajun
 * @date   2018年9月13日
 */
@Component
public class DeferredResultHolder {
	//map 其中key相当于订单号,一个订单号对应一个订单的处理结果
	private Map> map = new HashMap>();
	
	public Map> getMap() {
		return map;
	}
	
	public void setMap(Map> map) {
		this.map = map;
	}
}

AsyncController.java Controller控制器

package com.imooc.web.asnyc;

import java.util.concurrent.Callable;

import org.apache.commons.lang.RandomStringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;
import org.springframework.web.context.request.async.DeferredResult;

@RestController
public class AsyncController {
	
	@Autowired
	private MockQueue mockQueue;
	
	@Autowired
	private DeferredResultHolder deferredResultHolder;
	
	private Logger logger = LoggerFactory.getLogger(getClass());
	
	@RequestMapping("/order")
	public DeferredResult order() throws Exception {
		logger.info("主线程开启");
		//生产8位随机订单号
		String oderNumber = RandomStringUtils.randomNumeric(8);
		mockQueue.setPlaceOrder(oderNumber);
		//响应结果
		DeferredResult result = new DeferredResult<>();
		//注入
		deferredResultHolder.getMap().put(oderNumber, result);
		logger.info("主线程关闭");
		return result;
	}
}

QueueListener.java 监听消息队列是否已经完成

package com.imooc.web.asnyc;

import org.apache.commons.lang.StringUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.ApplicationListener;
import org.springframework.context.event.ContextRefreshedEvent;
import org.springframework.stereotype.Component;

/**
 * 监听虚拟消息队列   相当于线程二
 * @author caijiajun
 * @date   2018年9月13日
 */
@Component
public class QueueListener implements ApplicationListener{
	
	@Autowired
	private MockQueue mockQueue;
	
	@Autowired
	private DeferredResultHolder deferredResultHolder;
	
	private Logger logger = LoggerFactory.getLogger(getClass());
	
	//spring 加载完成后的事件
	@Override
	public void onApplicationEvent(ContextRefreshedEvent event) {
		
		new Thread(() -> {
			while (true) {
				//消息队列不为空
				if(StringUtils.isNotBlank(mockQueue.getCompleteOrder())) {
					String orderNumber = mockQueue.getCompleteOrder();
					logger.info("返回订单处理结果:" +orderNumber);
					deferredResultHolder.getMap().get(orderNumber).setResult("place order success");
					//还原消息队列为空
					mockQueue.setCompleteOrder(null);
				}else{
					try {
						Thread.sleep(100);
					} catch (InterruptedException e) {
						// TODO Auto-generated catch block
						e.printStackTrace();
					}
				}
			}
		}).start();
	}
	
	
}

在这里插入图片描述

  1. 上面的异步请求,都是模拟的,那么正在开发中的异步请求,spring是如何配置的呢?还是在WebMvcConfigurerAdapter的实现类里面进行配置
package com.imooc.web.config;

import java.util.ArrayList;
import java.util.List;

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.web.servlet.FilterRegistrationBean;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.AsyncSupportConfigurer;
import org.springframework.web.servlet.config.annotation.InterceptorRegistry;
import org.springframework.web.servlet.config.annotation.WebMvcConfigurerAdapter;

import com.imooc.filter.TimeFilter;
import com.imooc.web.interceptor.TimeInterceptor;

@Configuration
public class WebConfig extends WebMvcConfigurerAdapter{

	@Autowired
	private TimeInterceptor timeInterceptor;	
	
	/**
	 * 配置异步支持
	 */
	@Override
	public void configureAsyncSupport(AsyncSupportConfigurer configurer) {
		//配置对应的异步请求拦截器
		configurer.registerCallableInterceptors(timeInterceptor);//异步请求配置1、副线程就开在主线程里面
		configurer.registerDeferredResultInterceptors(timeInterceptor);//配置对应的异步请求拦截器:2、使用DeferredResult设置消息
		configurer.setDefaultTimeout(5000);//异步请求默认的超出时间
		configurer.setTaskExecutor(null);//设置可重用的线程池,来替代spring默认的不重用的线程池,我们上面的Callable看的线程是每次运行的时候都是开的一个新的线程。这种就是不重用的线程池。
	}
}

你可能感兴趣的:(Spring,Boot)