一、HTTP请求地址映射
1、请求地址匹配
@RequestMapping不但支持标准的URL,还支持Ant风格(即?、*和**的字符,参见3.3.2节的内容)的和带{xxx}占位符的URL。以下URL都是合法的:
/user/*/createUser
匹配/user/aaa/createUser、/user/bbb/createUser等URL。
/user/**/createUser
匹配/user/createUser、/user/aaa/bbb/createUser等URL。
/user/createUser??
匹配/user/createUseraa、/user/createUserbb等URL。
/user/{userId}
匹配user/123、user/abc等URL。
/user/**/{userId}
匹配user/aaa/bbb/123、user/aaa/456等URL。
company/{companyId}/user/{userId}/detail
匹配company/123/user/456/detail等的URL。
实例:
@RequestMapping("/a/{userId}")
public ModelAndView showDetail(@PathVariable("userId") String userId){
请求地址作为变量传输。
2、请求方法限定
@RequestMapping(value="/delete",method=RequestMethod.POST)
public String test1(@RequestParam("userId") String userId){
return "user/test1";
}
所有URL为<controllerURI>/delete 且请求方法为POST 的请求由test1处理
二、HTTP请求数据的绑定
1、@RequestParam?绑定请求参数
@RequestParam有以下三个参数。
value:参数名。
required:是否必需,默认为true,表示请求中必须包含对应的参数名,如果不存在将抛出异常。
defaultValue:默认参数名,设置该参数时,自动将required设为false。极少情况需要使用该参数,也不推荐使用该参数。
如果不能保证存在”userName”的参数,必须使用:@RequestParam(value = "userName", required = false) ,否则action会报错。
2、@CookieValue?绑定Cookie的值
3、@PathVariable?绑定URL中的变量
4、使用POJO对象绑定参数
POJO对象并不需要实现任何接口,仅是一个拥有若干属性的POJO。Spring MVC按:
“HTTP请求参数名 = POJO对象的属性名”
的规则,自动绑定请求数据,支持“级联属性名”,自动进行基本类型数据转换。
=====================================2012年12月4日
5、Servlet API对象做参数
public void handle21(HttpServletRequest request,HttpServletResponse response,HttpSession session) {
String name = request.getParameter("name");
response.addCookie(new Cookie("userName", userName));
session.setAttribute("sessionId", 1234);
}
6、Spring的Servlet API代理类做参数
@RequestMapping(value = "/handle25")
public String handle25(WebRequest request) {
String userName = request.getParameter("userName");
return "success";
}
7、使用IO对象作为入参
在页面上打印图片内容或者文件内容:
public void handle31(OutputStream os) throws IOException{
Resource res = new ClassPathResource("/db.properties");//读取类路径下的图片文件
FileCopyUtils.copy(res.getInputStream(), os);//将图片写到输出流中
}
8、HttpMessageConverter<T> http请求消息获取一般用不到
9、使用@RequestBody/@ResponseBody 请求和应答方法体
@ResponseBody是方法体return的东西直接作为HTTP response body 输出,而不是页面,跟response.getWriter().write()一样。
优点:处理方法签名灵活不受限
缺点:只能访问报文体,不能访问报文头
10、使用HttpEntity<T>/ResponseEntity<T>
优点:处理方法签名受限
缺点:不但可以访问报文体,还能访问报文头
11、输出JSON
搭配jackson的jar包,处理json得心应手
@RequestMapping("ajaxTest")
public String ajaxTest(HttpServletRequest request,HttpServletResponse response) throws Exception {
out = response.getWriter();//获取输出流
Student student = new Student();
student.setName("ajaxtest");
String str = mapper.writeValueAsString(student);
out.write(str);
return null;
}
三、 数据转换格式化、校验、国际化
1、数据转换
/**
* 作用于该Controller的类型转换
* 只要该方法即可,页面和pojo不需做处理
* @param binder
*/
@InitBinder
public void initBinder(WebDataBinder binder) {
// System.out.println("init binder =======================");
binder.registerCustomEditor(Date.class, new CustomDateEditor( new SimpleDateFormat("yyyy-MM-dd"), false));
binder.registerCustomEditor(Integer.class, null, new CustomNumberEditor(Integer.class, null, true));
binder.registerCustomEditor(Long.class, null, new CustomNumberEditor(Long.class, null, true));
binder.registerCustomEditor(Float.class, new CustomNumberEditor( Float.class, true));
binder.registerCustomEditor(Double.class, new CustomNumberEditor(Double.class, true));
binder.registerCustomEditor(BigInteger.class, new CustomNumberEditor(BigInteger.class, true));
}
2、校验
待续。。。
3、国际化(2012年12月6日)
。。。
====================
四、Sping MVC 处理请求
1、RequestMappingHandlerMapping会自动扫描@Controller 和@RequestMapping。
2、Interceptor拦截器:
Spring拦截器,集成HandlerInterceptorAdapter抽象类,该类实现了一个有3个方法的接口,3个方法:
preHandle处理前,postHandle处理后,afterCompletion请求完毕后。
示例:
preHandle返回值为boolean值,若为true执行继续。若为false,DispatcherServlet认为拦截器管理的请求,不再继续执行其他拦截器或者Controller。
待续。。。
拦截器配置:
<!-- 拦截器 -->
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/student/*"/>
<bean class="com.sunivo.framwork.demo.LoginInterceptor" />
</mvc:interceptor>
</mvc:interceptors>
4、 view resolvers视图处理
Srping可以配置多个view resolvers,并按照order属性开始找,DispatcherServlet将按照类型来获取ViewResolver。如果没有为某个ViewResolver指定order值的话,默认值为Integer.MAX_ VALUE,对应的是最低优先级。如果一个view resolver不能处理这种视图,则继续找下一个view revolver,直至处理或者报错。
========2012年12月6日
5、使用locales
A、AcceptHeaderLocaleResolver实现LocaleResolver接口,内部有2个简单方法,其中resolveLocale()方法返回一个request的Local信息,即return request.getLocale();
B、浏览器和服务器会话保持原理(插播一个知识)
浏览器访问服务器,服务器会检查请求头中的Cookie是否有JSESSIONID这一项,如果没有,服务器会创建一个Session对象,其ID属性为一个复杂字符串。并在应答时Set-Cookie,给浏览器发一个Cookie,name为JSESSIONID,value为这个Session对象的ID。
在浏览器的每次请求时,服务器仍然会检查Cookie的JSESSIONID这项,若发现了这一项,则服务器将ID为JSESSIONID值的Session对象与其匹配。
若Cookie的JSESSIONID值无法匹配服务器里的Session群,则服务器会重新创建一个Session对象并给于浏览器覆盖无法匹配的JSESSIONID Cookie.
C、CookieLocaleResolver和SessionLocaleResolver
感觉这些玩意不是用来处理国际化的。
6、定义themes主题
A、Springmvc支持将css,background图片等资源文件配置到一个properties 文件中,并在页面中做常量引入。这做到了如果想切换系统的主题,只要修改这个properties 文件即可做到切换。示例:
B、properties文件:
styleSheet=/themes/cool/style.css
background=/themes/cool/img/coolBg.jpg
C、页面引入:(springmvc的标签)
<%@ taglib prefix="spring" uri="http://www.springframework.org/tags"%>
<html>
<head>
<link rel="stylesheet" href="<spring:theme code='styleSheet'/>" type="text/css"/>
</head>
<body style="background=<spring:theme code='background'/>">
...
</body>
</html>
D、主题解析
在我们定义了主题以后,DispatcherServlet将会找到一个名字叫做themeResolver的主题解析器,主题解析器的工作原理和
LocaleResolver很类似。它检测对应一个请求应该使用的主题,并且可以动态的切换主题。下面是spring提供的主题解析器:
Class description
FixedThemeResolver 选择一个固定的主题,通过defaultThemeName属性设定
SessionThemeResolver 主题在用户的session中被管理,一个session只需设置一次,但不会被持久化。
CookieThemeResolver 主题选择被存储在客户端的cookie中
E、动态改变主题Theme resolvers
xml配置实例:
参数中设置themeName=XXX可以手动修改主题,通过下面的配置默认会找classthpath下面themes/cool.properties
<!-- theme解析器 -->
<bean id="themeChangeInterceptor"
class="org.springframework.web.servlet.theme.ThemeChangeInterceptor">
<property name="paramName" value="themeName"></property>
</bean>
<bean id="themeSource"
class="org.springframework.ui.context.support.ResourceBundleThemeSource">
<property name="basenamePrefix" value="themes/"></property>
</bean>
<bean id="themeResolver"
class="org.springframework.web.servlet.theme.CookieThemeResolver">
<property name="defaultThemeName" >
<value>cool</value>
</property>
</bean>