MVC开发

我们通过前面的章节可以看到:

  • Servlet适合编写Java代码,实现各种复杂的业务逻辑,但不适合输出复杂的HTML;
  • JSP适合编写HTML,并在其中插入动态内容,但不适合编写复杂的Java代码。

能否将两者结合起来,发挥各自的优点,避免各自的缺点?

答案是肯定的。我们来看一个具体的例子。

假设我们已经编写了几个JavaBean:

  1. public class User {
  2. public long id;
  3. public String name;
  4. public School school;
  5. }
  6. public class School {
  7. public String name;
  8. public String address;
  9. }

UserServlet中,我们可以从数据库读取UserSchool等信息,然后,把读取到的JavaBean先放到HttpServletRequest中,再通过forward()传给user.jsp处理:

  1. @WebServlet(urlPatterns = "/user")
  2. public class UserServlet extends HttpServlet {
  3. protected void doGet(HttpServletRequest req, HttpServletResponse resp) throws ServletException, IOException {
  4. // 假装从数据库读取:
  5. School school = new School("No.1 Middle School", "101 South Street");
  6. User user = new User(123, "Bob", school);
  7. // 放入Request中:
  8. req.setAttribute("user", user);
  9. // forward给user.jsp:
  10. req.getRequestDispatcher("/WEB-INF/user.jsp").forward(req, resp);
  11. }
  12. }

user.jsp中,我们只负责展示相关JavaBean的信息,不需要编写访问数据库等复杂逻辑:

  1. <%@ page import="com.itranswarp.learnjava.bean.*"%>
  2. <%
  3. User user = (User) request.getAttribute("user");
  4. %>
  5. <html>
  6. <head>
  7. <title>Hello World - JSP</title>
  8. </head>
  9. <body>
  10. <h1>Hello <%= user.name %>!</h1>
  11. <p>School Name:
  12. <span style="color:red">
  13. <%= user.school.name %>
  14. </span>
  15. </p>
  16. <p>School Address:
  17. <span style="color:red">
  18. <%= user.school.address %>
  19. </span>
  20. </p>
  21. </body>
  22. </html>

请注意几点:

  • 需要展示的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渲染:

mvc

我们把UserServlet看作业务逻辑处理,把User看作模型,把user.jsp看作渲染,这种设计模式通常被称为MVC:Model-View-Controller,即UserServlet作为控制器(Controller),User作为模型(Model),user.jsp作为视图(View),整个MVC架构如下:

  1. ┌───────────────────────┐
  2. ┌────>│Controller: UserServlet
  3. └───────────────────────┘
  4. ┌───────┐ ┌─────┴─────┐
  5. Browser│────┘ Model: User
  6. │<───┐ └─────┬─────┘
  7. └───────┘
  8. ┌───────────────────────┐
  9. └─────│ View: user.jsp
  10. └───────────────────────┘

使用MVC模式的好处是,Controller专注于业务处理,它的处理结果就是Model。Model可以是一个JavaBean,也可以是一个包含多个对象的Map,Controller只负责把Model传递给View,View只负责把Model给“渲染”出来,这样,三者职责明确,且开发更简单,因为开发Controller时无需关注页面,开发View时无需关心如何创建Model。

MVC模式广泛地应用在Web页面和传统的桌面程序中,我们在这里通过Servlet和JSP实现了一个简单的MVC模型,但它还不够简洁和灵活,后续我们会介绍更简单的Spring MVC开发。

练习

MVC开发 - 图2下载练习:使用MVC开发 (推荐使用IDE练习插件快速下载)

小结

MVC模式是一种分离业务逻辑和显示逻辑的设计模式,广泛应用在Web和桌面应用程序。

读后有收获可以支付宝请作者喝咖啡,读后有疑问请加微信群讨论:

MVC开发 - 图3MVC开发 - 图4