架构探险笔记11-与Servlet API解耦
Servlet API解耦
为什么需要与Servlet API解耦
目前在Controller中是无法调用Servlet API的,因为无法获取Request与Response这类对象,我们必须在Dispatcher中将这些对象传递给Controller的Action方法才能拿到这些对象,这显然会增加Controller对Servlet API的耦合。最好能让Controller完全不使用Servlet API就能操作Request与Response对象。
最容易拿到Request与Response对象的地方就是DispatcherServlet的service方法:
@WebServlet(urlPatterns = "/*",loadOnStartup = 0)
public class DispatcherServlet extends HttpServlet { @Override
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
...
}
}
然而,我们又不想把Request和Response对象传递到Controller的Action方法中,所以我们需要提供一个线程安全的对象,通过它来封装Request和Response对象,并提供一系列常用的Servlet API,这样我们就可以在Controller中随时通过该对象来操作Request与Response对象的方法了。需要强调的是,这个对象一定是线程安全的,也就是说每个请求线程独自拥有一份Request与Response对象,不同请求线程间是隔离的。
与Servlet API解耦的实现过程
一个简单的思路是,编写一个ServletHelper类,让它去封装Request与Response对象,提供常用的ServletAPI工具方法,并利用ThreadLocal技术来保证线程安全,代码如下:
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import javax.servlet.ServletContext;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import javax.servlet.http.HttpSession; public class ServletHelper {
private static final Logger LOGGER = LoggerFactory.getLogger(ServletHelper.class); /**
* 使每个线程独自拥有一份ServletHelper实例
*/
private static final ThreadLocal<ServletHelper> SERVLET_HELPER_HOLDER = new ThreadLocal<ServletHelper>(); private HttpServletRequest request;
private HttpServletResponse response; public ServletHelper(HttpServletRequest request, HttpServletResponse response) {
this.request = request;
this.response = response;
} /**
* 初始化
* @param request
* @param response
*/
public static void init(HttpServletRequest request,HttpServletResponse response){
SERVLET_HELPER_HOLDER.set(new ServletHelper(request,response));
} /**
* 销毁
*/
public static void destroy(){
SERVLET_HELPER_HOLDER.remove();
} /**
* 获取Request对象
* @return
*/
private static HttpServletRequest getRequest(){
return SERVLET_HELPER_HOLDER.get().request;
} /**
* 获取Response对象
* @return
*/
private static HttpServletResponse getResponse(){
return SERVLET_HELPER_HOLDER.get().response;
} /**
* 获取Session对象
* @return
*/
private static HttpSession getSession(){
return getRequest().getSession();
} /**
* 获取ServletContext对象
* @return
*/
private static ServletContext getContext(){
return getRequest().getServletContext();
}
}
最重要的就是init和destroy方法,我们需要在恰当的地方调用它们,哪里是最恰当的地方呢?当然是上面提到的DispatcherServlet的service方法。此外还提供了一系列私有的getter和setter方法,因为我们需要封装几个常用的Servlet API工具方法:
/**
* 将属性放入Request中
* @param key
* @param val
*/
public static void setRequestAttribute(String key,Object val){
getRequest().setAttribute(key,val);
} /**
* 获取Request中的属性
* @param key
* @param <T>
* @return
*/
public static <T> T getRequestAttribute(String key){
return (T) getRequest().getAttribute(key);
} /**
* 从Request中移除属性
* @param key
*/
public static void removeRequestAttribute(String key){
getRequest().removeAttribute(key);
} /**
* 重定向
* @param location
*/
public static void sendRedirect(String location){
try {
getResponse().sendRedirect(location);
} catch (IOException e) {
LOGGER.error("redirect failure",e);
}
} /**
* 将属性放入Session中
* @param key
* @param val
*/
public static void setSessionAttribute(String key,Object val){
getSession().setAttribute(key,val);
} /**
* 获取Session中的属性
* @param key
* @param <T>
* @return
*/
public static <T> T getSessionAttribute(String key){
return (T) getSession().getAttribute(key);
} /**
* 移除Session中的属性
* @param key
*/
public static void removeSessionAttribute(String key){
getSession().removeAttribute(key);
} /**
* 使Session失效
*/
public static void invalidateSession(){
getSession().invalidate();
}
以上这些工具方法都是可拓展的,只要是我们认为比较常用的都可以封装起来。
现在ServletHelper已经开发完毕,是时候将其整合到DispatcherServlet中并初始化Request与Response对象了,实际上就是调用init与destroy方法。
protected void service(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
ServletHelper.init(req,resp); //使每个线程都有独立的request和response
try {
/****/
}finally {
ServletHelper.destroy();
}
}
现在就可以在Controller类中随时调用ServletHelper封装的Servlet API了:而且不仅仅可以在Controller类中调用,实际上在Service类中也是可以调用。因为所有调用都来自同一请求线程。DispatcherServlet是请求线程的入口,随后请求线程会先后来到Controller与Service中,我们只需要使用ThreadLocal来确保ServletHelper对象中的Request与Response对象线程安全即可。
架构探险笔记11-与Servlet API解耦的更多相关文章
- Struts 2读书笔记-----Action访问Servlet API
Action访问Servlet API Struts2中的Action并没有和任何Servlet API耦合,这样框架更具灵活性,更易测试. 对于Web应用的控制器而言,不访问ServletAPI是几 ...
- Struts2笔记--Action访问Servlet API
Web应用中通常需要访问的Servlet API就是HttpServletRequest.HttpSession和ServletContext,这三个接口分别代表JSP内置对象中的request.se ...
- 架构探险笔记3-搭建轻量级Java web框架
MVC(Model-View-Controller,模型-视图-控制器)是一种常见的设计模式,可以使用这个模式将应用程序进行解耦. 上一章我们使用Servlet来充当MVC模式中的Controller ...
- 【Java EE 学习 35 上】【strus2】【类型转换器】【struts2和Servlet API解耦】【国际化问题】【资源文件乱码问题已经解决】
一.类型转换器 1.在动作类action中,声明和表单中name属性的值同名的属性,提供get和set方法,struts2就可以通过反射机制,从页面中获取对应的内容 package com.kdyzm ...
- 架构探险笔记12-安全控制框架Shiro
什么是Shiro Shiro是Apache组织下的一款轻量级Java安全框架.Spring Security相对来说比较臃肿. 官网 Shiro提供的服务 1.Authentication(认证) 2 ...
- 架构探险笔记6-ThreadLocal简介
什么是ThreadLocal? ThreadLocal直译为“线程本地”或“本地线程”,如果真的这么认为,那就错了!其实它就是一个容器,用于存放线程的局部变量,应该叫ThreadLocalVariab ...
- 架构探险笔记4-使框架具备AOP特性(上)
对方法进行性能监控,在方法调用时统计出方法执行时间. 原始做法:在内个方法的开头获取系统时间,然后在方法的结尾获取时间,最后把前后台两次分别获取的系统时间做一个减法,即可获取方法执行所消耗的总时间. ...
- 架构探险笔记5-使框架具备AOP特性(下)
开发AOP框架 借鉴SpringAOP的风格,写一个基于切面注解的AOP框架.在进行下面的步骤之前,确保已经掌了动态代理技术. 定义切面注解 /** * 切面注解 */ @Target(Element ...
- 读《架构探险——从零开始写Java Web框架》
内容提要 <架构探险--从零开始写Java Web框架>首先从一个简单的 Web 应用开始,让读者学会如何使用 IDEA.Maven.Git 等开发工具搭建 Java Web 应用:接着通 ...
随机推荐
- ICM Technex 2018 and Codeforces Round #463 (Div. 1 + Div. 2, combined) A
2018-02-19 A. Palindromic Supersequence time limit per test 2 seconds memory limit per test 256 mega ...
- 获取当前的日期时间 格式“yyyy-MM-dd HH:MM:SS”
function getNowFormatDate() { var date = new Date(); var seperator1 = "-"; var ...
- Django 创建项目笔记
基本命令 mkdir mysite # 创建项目目录,常取名mysite cd mysite virtualenv env # env\Scripts\activate.bat # Win pip i ...
- 彻底了解 suid, sgid ,sticky权限
sticky: 粘性的, 如 : sticky tape: 粘胶带 /tmp, /var/tmp: 位 sticky: 表示: 第一, 任何用户都可以在该目录下创建文件(编辑自己的文件),第二, 但是 ...
- shiro 前后端分离 seseeionId 问题
http://www.cnblogs.com/cshhs/p/9269411.html https://www.w3cschool.cn/shiro/rmvk1if1.html http://www. ...
- iframe初始化属性
<iframe id="user" src="xxx.html" frameborder="0" width="" ...
- URL简单梳理
# DEBUG模式: 开启debug模式后,修改项目代码时按下ctrl+s可重启项目: 项目中出现bug时,浏览器与控制台会打印错误信息: 在生产环境中禁止开启DEBUG模式,有很大的安全隐患: 将D ...
- sprinf sprintf_s 的用法
函数功能: 将数据格式化输出到字符串 函数原型: int sprintf( char *buffer, const char *format [,argument] ... ) 注意这里的buffer ...
- dplyr-高效的数据变换与整理工具--转载
1.背景简介 在数据分析工作中,经常需要对原始的数据集进行清洗.整理以及变换.常用的数据整理与变换工作主要包括:特定分析变量的选取.满足条件的数据记录的筛选.按某一个或几个变量排序.对原始变量进行加工 ...
- excel 获取当前日期
获取第几周(如第12周) ="第"&WEEKNUM(TODAY())&"周" 获取星期几(如星期一) =TEXT(TODAY(),"a ...