Spring MVC中常见的使用技巧说明


Spring MVC
博客分类: Spring

spring.

Table of Contents
Spring MVC

1. 簡介

Spring MVC是一個和Spring完美結合的Web框架。我先前學過Struts2,再學Spring MVC,覺得Spring MVC更為簡約、高效和優雅,它有不少東西借鉴自Struts2,如依賴注入等。它功能完整,推薦在新項目中替代Struts2,至於Struts,直接秒殺。

下面的配置均采用Spring 3版本,Spring MVC基於Java Servlet技術規範,主要處理流程圖:



來自瀏覽器的請求URL由DispatcherServlet統一處理(除了css、js、圖片等靜態文件),它符合Java Servlet規範。在web.xml文件中配置哪些URL由該Servlet處理。
DispatcherServlet根據配置(通常由註解或xml文件配置)將URL請求發往相應的Controller處理,這個過程會帶上服務器端傳遞過來的參數並自動裝載給Controller。
Controller接收到參數後,調用相應的方法處理之,方法裡可以調用業務層接口或數據層接口等,然後將處理結果打包到一個ModelAndView對象中並返回。
DispatcherServlet將ModelAndView對象發往視圖解析器viewResolver處理。
視圖解析器根據ModelAndView對象中指定的視圖模板位置和攜帶的數據渲染模板文件為html文件,返回渲染後的文件。
DispatcherServlet將html文件(有時候是純文本或多媒體文件)返回給瀏覽器。
很好的教程:點擊

2. 安裝及配置文件

建議使用eclipse或netbeans編寫程式,只需要導入Spring和Spring Web的所有jar包即可。

按照習慣,Spring的xml配置文件統一放在classpath根目錄下並遵循applicationContext-*.xml的命名規範;web.xml配置css、js、圖片文件為靜態文件,靠後綴識別,其它URL由DispatcherServlet處理。web.xml配置文件如下。

不要使用/js/*作為靜態文件的url-pattern,這樣可以直接訪問WEB-INF下的文件。

web.xml

xmlns:xsi="https://www.360docs.net/doc/c0920275.html,/2001/XMLSchema-instance"
xsi:schemaLocation="https://www.360docs.net/doc/c0920275.html,/xml/ns/javaee
https://www.360docs.net/doc/c0920275.html,/xml/ns/javaee/web-app_2_5.xsd">



characterEncodingFilter
org.springframework.web.filter.CharacterEncodingFilter

encoding
utf8


forceEncoding
true



characterEncodingFilter
/*




default
*.css


default

*.js


default
*.jpg


default
*.jpeg


default
*.png


default
*.gif


default
*.bmp




dispatcherServlet
org.springframework.web.servlet.DispatcherServlet

contextConfigLocation

classpath:applicationContext-*.xml

1


dispatcherServlet
/



index.jsp


Tomcat, Jetty, JBoss, and GlassFish 默認Servlet的名字 – “default”
Google App Engine 默認Servlet的名字 – “_ah_default”
Resin 默認Servlet的名字 – “resin-file”
WebLogic 默認Servlet的名字 – “FileServlet”
WebSphere 默認Servlet的名字 – “SimpleFileServlet”
classpath下關於Spring Web的配置文件:

applicationContext-web.xml

xmlns:xsi="https://www.360docs.net/doc/c0920275.html,/2001/XMLSchema-instance" xmlns:context="https://www.360docs.net/doc/c0920275.html,/schema/context"
xsi:schemaLocation="https://www.360docs.net/doc/c0920275.html,/schema/beans
https://www.360docs.net/doc/c0920275.html,/schema/beans/spring-beans-3.0.xsd
https://www.360docs.net/doc/c0920275.html,/schema/context
https://www.360docs.net/doc/c0920275.html,/schema/context/spring-context-3.0.xsd">












eans>
3. 編寫Controller之Hello World

3.1 編寫Controller

Controller可以是任意一個普通類,不需要繼承或實現任何東西。Controller類必須放在配置文件applicationContext-web.xml中指定的註解包下,如springmvc.web,應根據實際情況修改。

HelloWorldController.java
package springmvc.web;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;

// 通過註解使得該類變成Controller類
@Controller
// 根目錄,默認是"/",指定後為"/hello"
@RequestMapping("hello")
public class HelloWorldController {

// 該方法對應的URL,即訪問"/hello/hello_world"將調用該方法
@RequestMapping("hello_world")
public ModelAndView handleRequest() {

// modelAndView的名稱將會結合viewResolver配置的prefix和suffix合成一個頁面的路徑
ModelAndView modelAndView = new ModelAndView("hello_world");
// modelAndView對象攜帶的信息
modelAndView.addObject("message", "hello_world MVC!");
return modelAndView;
}

// 一個類中可以寫很多這樣的方法,名稱自取,使用註解映射到不同的URL

// 如果沒有附帶數據,也可以直接返回String類型的名稱
}
關於其它bean依賴注入,比如業務層或數據層的對象,使用註解可以讓Spring完成自動注入:

@Autowired
private CommonDao commonDao;
注意:ModelAndView的類不是org.springframework.web.portlet.ModelAndView

3.2 編寫jsp頁面

當請求鏈接”/hello/hello_world”發送到DispatcherServlet,轉發到HelloWorldController的handlerRequest方法,該方法返回modelAndView對象,根據該對象的名稱hello_world和配置文件中的prefix和suffix組合成物理jsp地址位置:/WEB-INF/jsp/hello_world.jsp

現在創建該jsp頁面:

hello_world.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib prefix="c" uri="https://www.360docs.net/doc/c0920275.html,/jsp/jstl/core" %>


這是Controller傳過來的數據message:${message}




modelAndView對象中附帶的message字符串將被渲染到jsp頁面中。

部署該標準Java Web項目到tomcat服務器上,本機訪問http://127.0.0.1:8080/spring_mvc/hello/hello_world就可以看到網頁,其中spring_mvc是我在eclipse中的項目名稱,默認是這樣:

这是Controller传过来的数据message:hello_world MVC!
4. 獲得來自瀏覽器的參數及環境變量

4.1 GET或POST傳遞的參數

Spring MVC採用控制反轉的思想進行依賴注入,假設HTML頁面需要傳遞參數name和age,則HTML頁面有個form:


Name:

Age:


這種提交形式是POST,也可以等同地使用GET方式提交:/some/u

rl?name=pugwoo&age=24

然後Controller中對應的方法只需要在參數上加上對應名稱和類型的參數:

@RequestMapping("/some/url")
public ModelAndView handleRequest(String name, Integer age) {
// 此時name和age已經擁有服務器端傳過來的值
// 類型也會自動轉換,比如從String轉成Integer

// 接下來,做你想做的事情
}
如果想指定採用GET或POST方式,在@RequestMapping中指定method即可:

@RequestMapping(value = "/url", method = RequestMethod.GET)

@RequestMapping(value = "/url", method = RequestMethod.POST)
注:我不喜欢Spring 2.0使用Spring MVC标签来写HTML页面的方式。雖然它提供了參數驗證功能,但這個由自己寫的js來做更好,這樣有利於開發的解耦。

類Struts2的OGNL傳參形式

當頁面的變量很多時,可以採用OGNL方式傳遞變量,加入有類Car,Car有屬性String name和String model並提供對應的getter和setter方法,則HTML的名稱可以寫為:”name”和”model”,然後在Controller的對應方法的參數中只寫Car car,Spring MVC會將該對象及其值構建好。

注意:的name屬性不是填“https://www.360docs.net/doc/c0920275.html,”,而是“name”。

Date類型的參數傳遞問題

頁面傳遞過來的數據均是String類型,爾後Spring將其轉化成對應類型。有些類型不是Spring默認可以轉換的,如Date。只需要在Controller.java代碼中加入一個binder方法即可:

@InitBinder
public void initBinder(WebDataBinder binder) {
SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd");
binder.registerCustomEditor(Date.class, new CustomDateEditor(dateFormat, true));
}
其中日期格式按照實際需要進行設置,如“yyyy-MM-dd HH:mm:ss”等。

4.2 獲得Cookie的值

在Controller對應的方法的參數使用@CookieValue註解獲得:

@RequestMapping("/some/url")
public ModelAndView handleRequest(@CookieValue("name") String name) {
// 接下來,做你想做的事情
}
4.3 上传文件

首先是HTML上傳頁面的form表格:





然後編寫Controller,注意一定要加上@RequestParam,原因未明:

UploadController.java
package springmvc.web;

import java.io.IOException;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestParam;
import org.springframework.web.multipart.MultipartFile;
import org.springframework.web.servlet.ModelAndView;

@Controller
public class UploadController {

@RequestMapping("upload")
// 一定要加上@RequestParam
public ModelAndView uploadFile(@RequestParam MultipartF

ile myfile) throws IOException{
ModelAndView modelAndView = new ModelAndView("result");

// 顯示文件內容
StringBuilder sb = new StringBuilder();

// 判斷文件是否已上傳
if (!myfile.isEmpty()) {
sb.append("文件名稱: " + myfile.getOriginalFilename() + "
");

String content = new String(myfile.getBytes());
content = content.replace("\n", "
");
sb.append(content);
} else {
sb.append("沒有選擇文件");
}

modelAndView.addObject("result", sb.toString());
return modelAndView;
}
}
然後新建一個/WEB-INF/result.jsp文件:

result.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>
<%@ taglib prefix="c" uri="https://www.360docs.net/doc/c0920275.html,/jsp/jstl/core" %>


上傳的文件信息:


${result}




如果出現https://www.360docs.net/doc/c0920275.html,ng.NoClassDefFoundError: org/apache/commons/io/output/DeferredFileOutputStream異常,加上https://www.360docs.net/doc/c0920275.html,mons.io包即可。

4.4 獲得環境變量

標準的Servlet規範中,有HttpServletRequest和HttpServletResponse等環節變量(關於這些變量的用途,請參見標準Servlet編程),Spring MVC可以很輕鬆地獲得,只需要像上面獲得參數一樣在方法參數中寫即可:

@RequestMapping("/some/url")
public ModelAndView handleRequest(HttpServletRequest request,
HttpServletResponse response) {

// 你已獲得HttpServletRequest和HttpServletResponse對象,比Struts2實現***Aware接口還簡單

// 接下來,做你想做的事情
}
5. REST形式的URL

REST形式的URL對搜索引擎友好,看起來也很簡潔漂亮。Spring MVC實現REST形式URL很簡單,例子:

/**
* 在RequestMapping中,变量用{}括起来
* 然后在处理方法中用@PathVariable获取,名称要一致
* @PathVariable 里面也可以加上value="xxx",名称可以不一致
*/
@RequestMapping("/{name}/page/{p}")
public void handleRequest(Writer writer, @PathVariable String name,
@PathVariable Integer p) throws IOException{
writer.write("you are reading " + name +"'s page " + p);
}
URL中的數值會被作為變量傳入到方法中,如name和p,同樣是依賴注入的思想,類型會自動轉換,訪問下面的URL查看效果:

http://127.0.0.1:8080/spring_mvc/pugwoo/page/1
http://127.0.0.1:8080/spring_mvc/pugwoo/page/2
http://127.0.0.1:8080/spring_mvc/nick/page/99
6. 其它技巧

輸出純文本

例子,支持中文:

@RequestMapping("test")
public void test(HttpServletResponse resp) throws IOException {
resp.setContentType("text/plain;charset=utf-8");
resp.getWriter().write("Test 測試中文");
}
該方法適合做cgi接口,輸出json等格式的數據。

客戶端跳轉

設定Controller對應的方法返回名稱“redirect:hello_world”即可跳轉到對應的頁面,如hello_world。浏览器的地址栏会变化。


服務器端跳轉

設定Controller對應的方法返回名稱“forward:hello_world”即可跳轉到對應的頁面,如hello_world。

forward並不能將變量通過modelAndView傳遞給新的URL,只能通過將變量放在request或session裏面來完成。而對於redirect,變量只能放在session裏面,不能放在request裏。

jsp頁面標籤jstl

導入標籤:

<%@ taglib prefix="c" uri="https://www.360docs.net/doc/c0920275.html,/jsp/jstl/core" %>
直接獲取某個變量的值:${message} 或 ${https://www.360docs.net/doc/c0920275.html,} 等

循環遍歷容器,假如有類Car,Car有屬性name並且提供getter方法,對象數組ArrayList cars,則java語句:

for(Car car : cars) {
System.out.println(car.getName());
}
等同於


${https://www.360docs.net/doc/c0920275.html,}


亂碼問題

上面的web.xml中已經解決了POST數據的亂碼問題,但是GET方式傳遞數據仍有亂碼,這是URL編碼的問題,需要在服務器容器端設置,以Tomcat為例,修改conf/server.xml文件對應位置新增“URIEncoding”:

connectionTimeout="20000"
redirectPort="8443" URIEncoding="utf-8"/>
修訂日誌

相关文档
最新文档