Spring MVC 完整示例
在本例中,我们将使用Spring MVC框架构建一个入门级web应用程序。Spring MVC 是Spring框架最重要的的模块之一。它以强大的Spring IoC容器为基础,并充分利用容器的特性来简化它的配置。
MVC框架是什么
模型-视图-控制器(MVC)是一个众所周知的以设计界面应用程序为基础的设计模式。它主要通过分离模型、视图及控制器在应用程序中的角色将业务逻辑从界面中解耦。通常,模型负责封装应用程序数据在视图层展示。视图仅仅只是展示这些数据,不包含任何业务逻辑。控制器负责接收来自用户的请求,并调用后台服务(manager或者dao)来处理业务逻辑。处理后,后台业务层可能会返回了一些数据在视图层展示。控制器收集这些数据及准备模型在视图层展示。MVC模式的核心思想是将业务逻辑从界面中分离出来,允许它们单独改变而不会相互影响。
在Spring MVC应用程序中,模型通常由POJO对象组成,它在业务层中被处理,在持久层中被持久化。视图通常是用JSP标准标签库(JSTL)编写的JSP模板。控制器部分是由dispatcher servlet负责,在本教程中我们将会了解更多它的相关细节。
一些开发人员认为业务层和DAO层类是MVC模型组件的一部分。我对此持有不同的意见。我不认为业务层及DAO层类为MVC框架的一部分。通常一个web应用是3层架构,即数据-业务-表示。MVC实际上是表示层的一部分。
Dispatcher Servlet(Spring控制器)
在最简单的Spring MVC应用程序中,控制器是唯一的你需要在Java web部署描述文件(即web.xml文件)中配置的Servlet。Spring MVC控制器 ——通常称作Dispatcher Servlet,实现了前端控制器设计模式。并且每个web请求必须通过它以便它能够管理整个请求的生命周期。
当一个web请求发送到Spring MVC应用程序,dispatcher servlet首先接收请求。然后它组织那些在Spring web应用程序上下文配置的(例如实际请求处理控制器和视图解析器)或者使用注解配置的组件,所有的这些都需要处理该请求。
在Spring3.0中定义一个控制器类,这个类必须标有@Controller注解。当有@Controller注解的控制器收到一个请求时,它会寻找一个合适的handler方法去处理这个请求。这就需要控制器通过一个或多个handler映射去把每个请求映射到handler方法。为了这样做,一个控制器类的方法需要被@RequestMapping注解装饰,使它们成为handler方法。
handler方法处理完请求后,它把控制权委托给视图名与handler方法返回值相同的视图。为了提供一个灵活的方法,一个handler方法的返回值并不代表一个视图的实现而是一个逻辑视图,即没有任何文件扩展名。你可以将这些逻辑视图映射到正确的实现,并将这些实现写入到上下文文件,这样你就可以轻松的更改视图层代码甚至不用修改请求handler类的代码。
为一个逻辑名称匹配正确的文件是视图解析器的责任。一旦控制器类已将一个视图名称解析到一个视图实现。它会根据视图实现的设计来渲染对应对象。
Spring入门示例
在这个应用程序中,我将创建最简单的员工管理应用程序的演示,它只有一个功能,即系统提供的所有雇员的列表。让我们记下此应用程序的目录结构。
现在让我们编写所有涉及的主要几个文件。
导入程序所需的依赖jar包。如上图所示。
web.xml
这最精简的web.xml文件声明了一个Servlet(即dispatcher servlet)来接收所有类型的请求。Dispatcher servlet在这里充当前端控制器的角色。
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" id="WebApp_ID" version="2.5"> <display-name>springmvc_employee</display-name> <!-- The front controller of this Spring Web application, responsible for handling all application requests --> <servlet> <servlet-name>springmvc</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:springmvc-servlet.xml</param-value> </init-param> <load-on-startup>1</load-on-startup> </servlet> <!-- Map all requests to the DispatcherServlet for handling --> <servlet-mapping> <servlet-name>springmvc</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping> </web-app>
spring-servlet.xml(你也可以用applicationContext.xml文件)
<?xml version="1.0" encoding="UTF-8"?> <beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context" xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.2.xsd"> <context:component-scan base-package="com.franson"></context:component-scan> <!-- configure the InternalResourceViewResolver --> <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver" id="internalResourceViewResolver"> <!-- 前缀 --> <property name="prefix" value="/WEB-INF/views/" /> <!-- 后缀 --> <property name="suffix" value=".jsp" /> </bean> </beans>
EmployeeController.java
package com.franson.controller; import javax.annotation.Resource; import org.springframework.stereotype.Controller; import org.springframework.ui.Model; import org.springframework.web.bind.annotation.RequestMapping; import org.springframework.web.bind.annotation.RequestMethod; import com.franson.service.EmployeeService; @Controller @RequestMapping(value="employee") public class EmployeeController { @Resource EmployeeService employeeService; @RequestMapping(value = "getall",method=RequestMethod.GET) public String getAllEmployees(Model model) { model.addAttribute("employees",employeeService.getAllEmployees()); return "employeeList"; } }
模型Employee.java
package com.franson.model; public class Employee { private String dept; private int id; private String name; public String getDept() { return dept; } public int getId() { return id; } public String getName() { return name; } public void setDept(String dept) { this.dept = dept; } public void setId(int id) { this.id = id; } public void setName(String name) { this.name = name; } @Override public String toString() { return "Employee [dept=" + dept + ", id=" + id + ", name=" + name + "]"; } }
IEmployeeDAO.java
这个类位于三层架构中的第三层。负责与底层的数据库存储进行交互。
package com.franson.dao; import java.util.List; import com.franson.model.Employee; public interface IEmployeeDao { List<Employee> getAllEmployees(); }
EmployeeDAO.java
package com.franson.dao; import java.util.ArrayList; import java.util.List; import org.springframework.stereotype.Repository; import com.franson.model.Employee; @Repository(value="defaultEmployeeDao") public class EmployeeDao implements IEmployeeDao { @Override public List<Employee> getAllEmployees() { List<Employee> lstEmployees = new ArrayList<Employee>(); Employee p1 = new Employee(); p1.setId(1); p1.setName("Franson"); p1.setDept("三所"); Employee p2 = new Employee(); p2.setId(2); p2.setName("Lily"); p2.setDept("一所"); Employee p3 = new Employee(); p3.setId(3); p3.setName("Tom"); p3.setDept("二所"); Employee p4 = new Employee(); p4.setId(4); p4.setName("Liao"); p4.setDept("五所"); lstEmployees.add(p1); lstEmployees.add(p2); lstEmployees.add(p3); lstEmployees.add(p4); return lstEmployees; } }
IEmployeeService.java
这个类处于三层架构中的第二层。负责与DAO层交互。
package com.franson.service; import java.util.List; import com.franson.model.Employee; public interface IEmployeeService { List<Employee> getAllEmployees(); }
具体实现类如下:
package com.franson.service; import java.util.List; import javax.annotation.Resource; import org.springframework.stereotype.Service; import com.franson.dao.IEmployeeDao; import com.franson.model.Employee; @Service public class EmployeeService implements IEmployeeService { @Resource(name = "defaultEmployeeDao") IEmployeeDao employeeDao; @Override public List<Employee> getAllEmployees() { return employeeDao.getAllEmployees(); } }
employeesList.jsp(结合bootstrap使用)
<%@ page language="java" contentType="text/html; charset=UTF-8" import="java.util.*" pageEncoding="UTF-8"%> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core"%> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>员工表</title> <!-- Bootstrap --> <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/css/bootstrap.min.css"> </head> <body> <div class="container"> <div class="row"> <table class="table table-striped"> <tr> <th>ID</th> <th>用户名</th> <th>所在部门</th> </tr> <c:forEach items="${employees}" var="employ"> <tr> <td>${employ.id}</td> <td>${employ.name}</td> <td>${employ.dept}</td> </tr> </c:forEach> </table> </div> </div> <script src="//cdn.bootcss.com/jquery/1.11.3/jquery.min.js" /> <script src="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.6/js/bootstrap.min.js" /> </body> </html>
部署于tomcat6容器中,在浏览器中输入地址:http://localhost:8080/springmvc_employee/mvcemployee/all
可看到如下图所示的结果: