https://docs.spring.io/spring/docs/5.0.9.RELEASE/spring-framework-reference/web.html
- @Bean 指定初始化和销毁方法
- 实现 InitializingBean和DisposableBean 指定初始化和销毁方法
- @PostConstruct / @PreDestroy 指定初始化和销毁方法
- @Autowired / @Qualifier / @Primary
- @Resource / @Inject
- 实现 Aware 注入 Spring 底层组件
- @Profile 环境搭建
- @Profile 环境激活
- @Aspect / @Pointcut / @Before / @After / @AfterReturning / @AfterThrowing / @Around
- @EnableAspectJAutoProxy
组件加载流程:
-
导入 webmvc 包后,lib 中多了一个
spring-web-4.3.12.RELEASE.jar
-
web容器在启动的时候,会扫描每个jar包下的 META-INF/services/javax.servlet.ServletContainerInitializer 。此时在
spring-web-4.3.12.RELEASE.jar
包中扫描到了该文件: -
加载文件中指定的类:
org.springframework.web.SpringServletContainerInitializer
-
SpringServletContainerInitializer 会加载 WebApplicationInitializer 接口下所有的组件。并且为 WebApplicationInitializer 组件创建对象(当组件不是接口,不是抽象类时)
我们的 web配置类
要继承的类: AbstractAnnotationConfigDispatcherServletInitializer
是 WebApplicationInitializer 接口的间接子类。
public class MyWebAppInitializer extends AbstractAnnotationConfigDispatcherServletInitializer {
/**
* 加载根容器的配置类:Spring配置
*/
@Override
protected Class<?>[] getRootConfigClasses() {
return new Class[]{RootConfig.class};
}
/**
* 加载webmvc容器的配置类:SpringMVC 配置
*/
@Override
protected Class<?>[] getServletConfigClasses() {
return new Class[]{MvcConfig.class};
}
/**
* 获取 DispatcherServlet 的映射信息
* <p>
* - `/` 拦截所有请求(包括静态资源(xx.js,xx.png)),但是不包括*.jsp;
* - `/*` 拦截所有请求;连*.jsp页面都拦截;jsp页面是tomcat的jsp引擎解析的;
*/
@Override
protected String[] getServletMappings() {
return new String[]{"/"};
}
}
@ComponentScan(value = {"xyz.yuzh.learn.spring.annotation"}, includeFilters = {
@ComponentScan.Filter(type = FilterType.ANNOTATION, classes = Controller.class)
}, useDefaultFilters = false)
@EnableWebMvc // 开启 SpringMVC 定制配置功能,等同于 <mvc:annotation-driven/>
public class WebConfig extends WebMvcConfigurerAdapter { // WebMvcConfigurerAdapter抽象类 实现了 WebMvcConfigurer 接口来定制MVC的所有配置,实现抽象类使得我们不必重写所有所有接口方法
/**
* 定制视图解析器
*/
@Override
public void configureViewResolvers(ViewResolverRegistry registry) {
registry.jsp("/WEB-INF/views/", ".jsp"); // 若不加参数默认为:("/WEB-INF/",".jsp")
}
/**
*controller
* springMVC 默认拦截所有的请求包括静态资源,解除静态资源访问限制,但仅对WEB根目录下文件有效,如果文件在WEB-INF解除无效。
*/
/*@Override
public void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
configurer.enable();
}*/
/**
* 定制拦截器
*/
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(new MyIntercepter()).addPathPatterns("/**"); // 拦截所有目录及文件
}
/**
* 这种方式访问 WEB-INF 下的静态资源 <mvc:resources location="/WEB-INF/resources/" mapping="/resources/**"/>
*/
@Override
public void addResourceHandlers(ResourceHandlerRegistry registry) {
registry.addResourceHandler("/resources/**").addResourceLocations("/WEB-INF/resources/");
}
}
更多配置查阅 文档
在Servlet 3.0之前,Servlet采用Thread-Per-Request的方式处理请求。即每一次Http请求都由某一个线程从头到尾负责处理。
如果一个请求需要进行IO操作,比如访问数据库、调用第三方服务接口等,那么其所对应的线程将同步地等待IO操作完成, 而IO操作是非常慢的,所以此时的线程并不能及时地释放回线程池以供后续使用,在并发量越来越大的情况下,当线程池已满,许多请求不能及时的被处理。这将带来严重的性能问题。
即便是像Spring、Struts这样的高层框架也脱离不了这样的桎梏,因为他们都是建立在Servlet之上的。为了解决这样的问题,Servlet 3.0引入了异步处理,然后在Servlet 3.1中又引入了非阻塞IO来进一步增强异步处理的性能。
@WebServlet(value = "/async", asyncSupported = true) // 支持异步处理asyncSupported=true
public class HelloAsyncServlet extends HttpServlet {
@Override
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 开启异步模式
System.out.println("主线程开始。。。" + Thread.currentThread() + "->" + System.currentTimeMillis());
AsyncContext startAsync = req.startAsync();
//业务逻辑进行异步处理; 开始异步处理
startAsync.start(new Runnable() {
@Override
public void run() {
try {
System.out.println("副线程开始 " + Thread.currentThread() + "->" + System.currentTimeMillis());
sayHello();
startAsync.complete(); // 异步处理完
AsyncContext asyncContext = req.getAsyncContext(); // 获取到异步上下文
ServletResponse response = asyncContext.getResponse(); // 获取响应
response.getWriter().write("hello async...");
System.out.println("副线程结束 " + Thread.currentThread() + "->" + System.currentTimeMillis());
} catch (Exception e) {
}
}
});
System.out.println("主线程结束。。。" + Thread.currentThread() + "->" + System.currentTimeMillis());
}
public void sayHello() throws Exception {
System.out.println(Thread.currentThread() + " processing...");
Thread.sleep(3000);
}
}
基于 Callable
的简单异步处理。返回 Callable 结果对象,将 Callable 对象放入异步处理线程处理结果,主线程结束,等待异步线程返回结果渲染页面。
@Controller
public class AsyncController {
/**
* 异步处理
* 1. 控制器返回 Callable
* 2. Spring 异步处理,将Callable 提交到 TaskExecutor 使用一个隔离的线程进行执行
* 3. DispatcherServlet 和所有的 Filter 退出 web 容器的线程,但是 response 保持打开状态;
* 4. Callable 返回结果,Spring MVC 将请求重新派发给容器,恢复之前的处理;
* 5. 根据 Callable 返回的结果。Spring MVC 继续进行视图渲染流程等(从收请求-视图渲染)。
*/
@ResponseBody
@RequestMapping("/async01")
public Callable<String> async01() {
System.out.println("主线程开始." + Thread.currentThread() + "->" + System.currentTimeMillis());
Callable<String> callable = new Callable<String>() {
@Override
public String call() throws Exception {
System.out.println("副线程开始." + Thread.currentThread() + "->" + System.currentTimeMillis());
Thread.sleep(2000);
System.out.println("副线程开始." + Thread.currentThread() + "->" + System.currentTimeMillis());
return "Callable<String> async01()";
}
};
System.out.println("主线程结束." + Thread.currentThread() + "->" + System.currentTimeMillis());
return callable;
}
}
基于 DeferredResult
的异步处理。这种方式的特点在于可以从不同的线程异步地产生返回值,配合 JMS ,可以实现在其他控制器中返回本次请求的结果。
@GetMapping("/quotes")
@ResponseBody
public DeferredResult<String> quotes() {
DeferredResult<String> deferredResult = new DeferredResult<String>();
// 把这个对象存在一个地方
return deferredResult;
}
// 在某个地方获取对象并存入对象的值然后返回
deferredResult.setResult(data);
用例:
@Controller
public class AsyncController {
/**************************************************
* DeferredResult
**************************************************/
/**
* 处理请求:创建订单
* - 1. 构建 DeferredResult 对象,设置超时事件,失败提示
* - 2. 通过 JMS 发送一个 DeferredResultQueue 对象的消息,该对象存储 DeferredResult.
* - 3. 若接收端在限定时间内给 DeferredResult 存入了结果,则返回 DeferredResult 的结果,否则结果是失败提示。
*/
@ResponseBody
@RequestMapping("/createOrder")
public DeferredResult<Object> createOrder() {
DeferredResult<Object> deferredResult = new DeferredResult<>((long) 3000, "create fail...");
// 模拟发送消息:把结果对象存入一个队列中,假设 DeferredResultQueue 队列就是消息通知内容
DeferredResultQueue.save(deferredResult);
// 等待异步存入结果到 deferredResult 中并返回,超时则失败
return deferredResult;
}
/**
* 模拟异步的存入请求结果:执行创建
* - 1. 监听 JMS 消息,获得消息内容(DeferredResultQueue对象)
* - 2. 从消息对象 DeferredResultQueue 中拿到 DeferredResult 延迟异步结果对象。
* - 3. 执行业务逻辑,赋值。
*/
@ResponseBody
@RequestMapping("/create")
public String create() {
//创建订单
String order = UUID.randomUUID().toString();
// 获取到消息通知,从中获取到需要存取结果的对象
DeferredResult<Object> deferredResult = DeferredResultQueue.get();
// 向结果对象存入值,请求返回结果
deferredResult.setResult(order);
return "success: " + order;
}
}
class DeferredResultQueue {
private static ConcurrentLinkedQueue<DeferredResult<Object>> queue = new ConcurrentLinkedQueue<DeferredResult<Object>>();
public static void save(DeferredResult<Object> deferredResult) {
queue.add(deferredResult);
}
public static DeferredResult<Object> get() {
return queue.poll();
}
}
-
Web: WebInitializer
spring context / spring mvc context / postFilter ...
-
Spring:RootConfig
dataSource / transactionManager / sqlSessionFactory ...
-
Spring MVC: DispatcherServletConfig
viewResolvers / resourceHandlers ...