MVC
什么是 MVC 框架
假设我们已经编写了几个JavaBean:
public class User {
public long id;
public String name;
public School school;
}
public class School {
public String name;
public String address;
}在 UserServlet 中,我们可以从数据库读取 User、School 等信息,然后,把读取到的 JavaBean 先放到 HttpServletRequest 中,再通过 forward() 传给 user.jsp 处理:
@WebServlet(urlPatterns = "/user")
public class UserServlet extends HttpServlet {
protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
// 假装从数据库读取:
School school = new School("No.1 Middle School", "101 South Street");
User user = new User(123, "Bob", school);
// 放入Request中:
req.setAttribute("user", user);
// forward给user.jsp:
req.getRequestDispatcher("/WEB-INF/user.jsp").forward(req, resp);
}
}在 user.jsp 中,我们只负责展示相关 JavaBean 的信息,不需要编写访问数据库等复杂逻辑:
需要展示的 User 被放入
HttpServletRequest中以便传递给 JSP,因为一个请求对应一个 HttpServletRequest,我们也无需清理它,处理完该请求后HttpServletRequest实例将被丢弃;把
user.jsp放到/WEB-INF/目录下,是因为WEB-INF是一个特殊目录,Web Server 会阻止浏览器对WEB-INF目录下任何资源的访问,这样就防止用户通过/user.jsp路径直接访问到 JSP 页面;JSP 页面首先从
request变量获取User实例,然后在页面中直接输出,此处未考虑 HTML 的转义问题,有潜在安全风险。
在浏览器访问 http://localhost:8080/user,请求首先由 UserServlet 处理,然后交给 user.jsp 渲染
我们把 UserServlet 看作业务逻辑处理,把 User 看作模型,把 user.jsp 看作渲染,这种设计模式通常被称为 MVC:Model-View-Controller,即 UserServlet 作为控制器(Controller),User 作为模型(Model),user.jsp 作为视图(View),整个 MVC 架构如下:
使用 MVC 模式的好处是,Controller 专注于业务处理,它的处理结果就是 Model。Model 可以是一个 JavaBean,也可以是一个包含多个对象的 Map,Controller 只负责把 Model 传递给 View,View 只负责把 Model 给 “渲染” 出来,这样,三者职责明确,且开发更简单,因为开发 Controller 时无需关注页面,开发 View 时无需关心如何创建 Model。
MVC高级开发
通过结合 Servlet 和 JSP 的 MVC 模式,我们可以发挥二者各自的优点:
Servlet 实现业务逻辑;
JSP 实现展示逻辑。
但是,直接把 MVC 搭在 Servlet 和 JSP 之上还是不太好,原因如下:
Servlet 提供的接口仍然偏底层,需要实现 Servlet 调用相关接口;
JSP 对页面开发不友好,更好的替代品是模板引擎;
业务逻辑最好由纯粹的 Java 类实现,而不是强迫继承自 Servlet。
能不能通过普通的 Java 类实现 MVC 的 Controller?类似下面的代码:
上面的这个 Java 类每个方法都对应一个 GET 或 POST 请求,方法返回值是 ModelAndView,它包含一个 View 的路径以及一个 Model,这样,再由 MVC 框架处理后返回给浏览器。
如果是 GET 请求,我们希望 MVC 框架能直接把 URL 参数按方法参数对应起来然后传入:
如果是 POST 请求,我们希望 MVC 框架能直接把 Post 参数变成一个 JavaBean 后通过方法参数传入:
为了增加灵活性,如果 Controller 的方法在处理请求时需要访问 HttpServletRequest、HttpServletResponse、HttpSession 这些实例时,只要方法参数有定义,就可以自动传入:
以上就是我们在设计 MVC 框架时,上层代码所需要的一切信息。
设计 MVC 框架
如何设计一个 MVC 框架?在上文中,我们已经定义了上层代码编写 Controller 的一切接口信息,并且并不要求实现特定接口,只需返回 ModelAndView 对象,该对象包含一个 View 和一个 Model。实际上 View 就是模板的路径,而 Model 可以用一个 Map<String, Object> 表示,因此,ModelAndView 定义非常简单:
比较复杂的是我们需要在 MVC 框架中创建一个接收所有请求的 Servlet,通常我们把它命名为 DispatcherServlet,它总是映射到 /,然后,根据不同的 Controller 的方法定义的 @Get 或 @Post 的 Path 决定调用哪个方法,最后,获得方法返回的 ModelAndView 后,渲染模板,写入 HttpServletResponse,即完成了整个 MVC 的处理。
这个 MVC 的架构如下:
其中,DispatcherServlet 以及如何渲染均由 MVC 框架实现,在 MVC 框架之上只需要编写每一个 Controller。
我们来看看如何编写最复杂的 DispatcherServlet。首先,我们需要存储请求路径到某个具体方法的映射:
处理一个 GET 请求是通过 GetDispatcher 对象完成的,它需要如下信息:
有了以上信息,就可以定义 invoke() 来处理真正的请求:
上述代码比较繁琐,但逻辑非常简单,即通过构造某个方法需要的所有参数列表,使用反射调用该方法后返回结果。
类似的,PostDispatcher 需要如下信息:
和 GET 请求不同,POST 请求严格地来说不能有 URL 参数,所有数据都应当从 Post Body 中读取。这里我们为了简化处理,只支持 JSON 格式的 POST 请求,这样,把 Post 数据转化为 JavaBean 就非常容易。
最后,我们来实现整个 DispatcherServlet 的处理流程,以 doGet() 为例:
这里有几个小改进:
允许 Controller 方法返回
null,表示内部已自行处理完毕;允许 Controller 方法返回以
redirect: 开头的 view 名称,表示一个重定向。
这样使得上层代码编写更灵活。例如,一个显示用户资料的请求可以这样写:
最后一步是在 DispatcherServlet 的 init() 方法中初始化所有 Get 和 Post 的映射,以及用于渲染的模板引擎:
如何扫描所有 Controller 以获取所有标记有 @GetMapping 和 @PostMapping 的方法?当然是使用反射了。
这样,整个 MVC 框架就搭建完毕。
实现渲染
如何使用模板引擎进行渲染有疑问,即如何实现上述的 ViewEngine?其实 ViewEngine 非常简单,只需要实现一个简单的 render() 方法:
Java 有很多开源的模板引擎,常用的有:
Thymeleaf
FreeMarker
Velocity
他们的用法都大同小异。这里我们推荐一个使用 Jinja 语法的模板引擎 Pebble,它的特点是语法简单,支持模板继承,编写出来的模板类似:
即变量用 {{ xxx }} 表示,控制语句用 `
` 表示。
使用 Pebble 渲染只需要如下几行代码:
最后我们来看看整个工程的结构:
其中,framework 包是 MVC 的框架,完全可以单独编译后作为一个 Maven 依赖引入,controller 包才是我们需要编写的业务逻辑。
我们还硬性规定模板必须放在 webapp/WEB-INF/templates 目录下,静态文件必须放在 webapp/static 目录下,因此,为了便于开发,我们还顺带实现一个 FileServlet 来处理静态文件:
运行代码,在浏览器中输入 URL http://localhost:8080/hello?name=Bob 可以看到如下页面:
为了把方法参数的名称编译到 class 文件中,以便处理 @GetMapping 时使用,我们需要打开编译器的一个参数,在 Eclipse 中勾选 Preferences-Java-Compiler-Store information about method parameters (usable via reflection);在 Idea 中选择 Preferences-Build, Execution, Deployment-Compiler-Java Compiler-Additional command line parameters,填入 - parameters;在 Maven 的 pom.xml 添加一段配置如下:
本次实现的这个 MVC 框架,上层代码使用的公共类如 GetMapping、PostMapping 和 ModelAndView 都和 Spring MVC 非常类似。实际上,我们这个 MVC 框架主要参考就是 Spring MVC,通过实现一个 “简化版”MVC,可以掌握 Java Web MVC 开发的核心思想与原理,对将来直接使用 Spring MVC 是非常有帮助的。
Source & Reference
https://www.liaoxuefeng.com/wiki/1252599548343744/1266264917931808
https://www.liaoxuefeng.com/wiki/1252599548343744/1337408645759009