- 浏览: 202712 次
- 性别:
- 来自: 上海
文章分类
最新评论
Spring 注解学习手札(二) 控制层梳理
言归正传,研究一下注解下的控制层。
我习惯于使用JSTL展示页面,因此需要在原lib基础上增加jstl.jar和standard.jar,详细lib依赖如下:
aopalliance-1.0.jar
commons-logging-1.1.1.jar
log4j-1.2.15.jar
spring-beans-2.5.6.jar
spring-context-2.5.6.jar
spring-context-support-2.5.6.jar
spring-core-2.5.6.jar
spring-tx-2.5.6.jar
spring-web-2.5.6.jar
spring-webmvc-2.5.6.jar
standard.jar
jstl.jar
上一篇文中,我们定义了控制器AccountController:
AccountController.java
先说注解@RequestMapping
这里使用注解@RequestMapping(method = RequestMethod.GET)指定这个方法为get请求时调用。同样,我们可以使用注解@RequestMapping(method = RequestMethod.POST)指定该方法接受post请求。
这与我们久别的Servlet很相像,类似于doGet()和doPost()方法!
我们也可以将其改造为多动作控制器,如下代码所示:
这样,我们可以通过参数“method”指定不同的参数值从而通过请求("/account.do?method=login"和"/account.do?method=logout")调用不同的方法!
注意:使用多动作控制器必须在配置文件中加入注解支持!
当然,我们也可以将注解@RequestMapping指定到某一个方法上,如:
这样,请求“a.do”和“b.do”将对应不同的方法a() 和b()。这使得一个控制器可以同时承载多个请求!
@RequestMapping("/account.do")是@RequestMapping(value="/account.do")的简写!
再说输入参数!
这里的方法名可以随意定义,但是参数和返回值却又要求!
为什么?直接看源代码,我们就能找到答案!
AnnotationMethodHandlerAdapter.java部分源代码——有关参数部分:
也就是说,如果我们想要在自定义的方法中获得一些个“标准”输入参数,参数类型必须包含在以下类型中:
ServletRequest
ServletResponse
HttpSession
Principal
Locale
InputStream
OutputStream
Reader
Writer
当然,上述接口其实都是对于HttpServletRequest和HttpServletResponse的扩展。
此外,我们还可以定义自己的参数。
注意:自定义参数必须是实现类,绝非接口!Spring容器将帮你完成对象初始化工作!
比如说上文中,我们需要参数username和password。我们可以这么写:
如果参数名不能与这里的变量名保持一致,那么我们可以使用注解@RequestParam进行强制绑定,代码如下所示:
这比起我们之前写的代码有所简洁:
ServletRequestUtils类的工作已经由Spring底层实现了,我们只需要把参数名定义一致即可,其内部取参无需关心!
除了传入参数,我们还可以定义即将传出的参数,如加入ModelMap参数:
这时,我们没有定义页面名称,Spring容器将根据请求名指定同名view,即如果是jap页面,则account.do->account.jsp!
不得不承认,这样写起来的确减少了代码量!
接着说输出参数!
通过ModelMap,我们可以绑定输出到的页面的参数,但最终我们将要返回到何种页面呢?再次查看AnnotationMethodHandlerAdapter源代码!
AnnotationMethodHandlerAdapter.java部分源代码——有关返回值部分:
返回值的定义十分庞大,或者说可怕的if-else多少有点让我觉得厌恶!
我们可以定义以下类型的返回值:
ModelAndView
Model
View
Map
String
null
ModelAndView、Model和View都是Spring之前版本所特有的元素,Map对应于传入参数ModelMap,String定义页面名称,null即对应void类型方法!
最常用的实现方式如下:
当然,对于我来说在返回值中写入这么一个字符串多少有点不能接受,于是我还是乐于使用输入参数ModelMap+输出参数Map的方式。
给出一个完整的AccountController实现:
AccountController.java
最后说注解@Session
如果想将某个ModelMap中的参数指定到Session中,可以使用@Session注解,将其绑定为Session熟悉,代码如下所示:
当然,我们还需要配置一下对应的视图解析器,给出完整配置:
servelt.xml
这里使用了JstlView作为视图解析器。同时,指定前缀路径为"/WEB-INF/page/",后缀路径为".jsp"。也就是说,Spring容器将会在这个路径中寻找匹配的jsp文件!
注意加入xmlns:p="http://www.springframework.org/schema/p"命名空间!
再给出页面内容:
taglib.jsp
account.jap
目录结构如图所示:
启动应用,最后将得到一个带有内容的页面,如图:
代码见附件!
我习惯于使用JSTL展示页面,因此需要在原lib基础上增加jstl.jar和standard.jar,详细lib依赖如下:
引用
aopalliance-1.0.jar
commons-logging-1.1.1.jar
log4j-1.2.15.jar
spring-beans-2.5.6.jar
spring-context-2.5.6.jar
spring-context-support-2.5.6.jar
spring-core-2.5.6.jar
spring-tx-2.5.6.jar
spring-web-2.5.6.jar
spring-webmvc-2.5.6.jar
standard.jar
jstl.jar
上一篇文中,我们定义了控制器AccountController:
AccountController.java
- /**
- *2010-1-23
- */
- packageorg.zlex.spring.controller;
- importjavax.servlet.http.HttpServletRequest;
- importjavax.servlet.http.HttpServletResponse;
- importorg.springframework.beans.factory.annotation.Autowired;
- importorg.springframework.stereotype.Controller;
- importorg.springframework.web.bind.ServletRequestUtils;
- importorg.springframework.web.bind.annotation.RequestMapping;
- importorg.springframework.web.bind.annotation.RequestMethod;
- importorg.zlex.spring.service.AccountService;
- /**
- *
- *@author<ahref="mailto:zlex.dongliang@gmail.com">梁栋</a>
- *@version1.0
- *@since1.0
- */
- @Controller
- @RequestMapping("/account.do")
- publicclassAccountController{
- @Autowired
- privateAccountServiceaccountService;
- @RequestMapping(method=RequestMethod.GET)
- publicvoidhello(HttpServletRequestrequest,HttpServletResponseresponse)
- throwsException{
- Stringusername=ServletRequestUtils.getRequiredStringParameter(
- request,"username");
- Stringpassword=ServletRequestUtils.getRequiredStringParameter(
- request,"password");
- System.out.println(accountService.verify(username,password));
- }
- }
先说注解@RequestMapping
这里使用注解@RequestMapping(method = RequestMethod.GET)指定这个方法为get请求时调用。同样,我们可以使用注解@RequestMapping(method = RequestMethod.POST)指定该方法接受post请求。
- @Controller
- @RequestMapping("/account.do")
- publicclassAccountController{
- @RequestMapping(method=RequestMethod.GET)
- publicvoidget(){
- }
- @RequestMapping(method=RequestMethod.POST)
- publicvoidpost(){
- }
- }
这与我们久别的Servlet很相像,类似于doGet()和doPost()方法!
我们也可以将其改造为多动作控制器,如下代码所示:
- @Controller
- @RequestMapping("/account.do")
- publicclassAccountController{
- @RequestMapping(params="method=login")
- publicvoidlogin(){
- }
- @RequestMapping(params="method=logout")
- publicvoidlogout(){
- }
这样,我们可以通过参数“method”指定不同的参数值从而通过请求("/account.do?method=login"和"/account.do?method=logout")调用不同的方法!
注意:使用多动作控制器必须在配置文件中加入注解支持!
当然,我们也可以将注解@RequestMapping指定到某一个方法上,如:
- @Controller
- publicclassAccountController{
- @RequestMapping("/a.do")
- publicvoida(){}
- @RequestMapping("/b.do")
- publicvoidb(){}
- }
这样,请求“a.do”和“b.do”将对应不同的方法a() 和b()。这使得一个控制器可以同时承载多个请求!
@RequestMapping("/account.do")是@RequestMapping(value="/account.do")的简写!
再说输入参数!
这里的方法名可以随意定义,但是参数和返回值却又要求!
为什么?直接看源代码,我们就能找到答案!
AnnotationMethodHandlerAdapter.java部分源代码——有关参数部分:
- @Override
- protectedObjectresolveStandardArgument(ClassparameterType,NativeWebRequestwebRequest)
- throwsException{
- HttpServletRequestrequest=(HttpServletRequest)webRequest.getNativeRequest();
- HttpServletResponseresponse=(HttpServletResponse)webRequest.getNativeResponse();
- if(ServletRequest.class.isAssignableFrom(parameterType)){
- returnrequest;
- }
- elseif(ServletResponse.class.isAssignableFrom(parameterType)){
- this.responseArgumentUsed=true;
- returnresponse;
- }
- elseif(HttpSession.class.isAssignableFrom(parameterType)){
- returnrequest.getSession();
- }
- elseif(Principal.class.isAssignableFrom(parameterType)){
- returnrequest.getUserPrincipal();
- }
- elseif(Locale.class.equals(parameterType)){
- returnRequestContextUtils.getLocale(request);
- }
- elseif(InputStream.class.isAssignableFrom(parameterType)){
- returnrequest.getInputStream();
- }
- elseif(Reader.class.isAssignableFrom(parameterType)){
- returnrequest.getReader();
- }
- elseif(OutputStream.class.isAssignableFrom(parameterType)){
- this.responseArgumentUsed=true;
- returnresponse.getOutputStream();
- }
- elseif(Writer.class.isAssignableFrom(parameterType)){
- this.responseArgumentUsed=true;
- returnresponse.getWriter();
- }
- returnsuper.resolveStandardArgument(parameterType,webRequest);
- }
也就是说,如果我们想要在自定义的方法中获得一些个“标准”输入参数,参数类型必须包含在以下类型中:
引用
ServletRequest
ServletResponse
HttpSession
Principal
Locale
InputStream
OutputStream
Reader
Writer
当然,上述接口其实都是对于HttpServletRequest和HttpServletResponse的扩展。
此外,我们还可以定义自己的参数。
注意:自定义参数必须是实现类,绝非接口!Spring容器将帮你完成对象初始化工作!
比如说上文中,我们需要参数username和password。我们可以这么写:
- @RequestMapping(method=RequestMethod.GET)
- publicvoidhello(Stringusername,Stringpassword){
- System.out.println(accountService.verify(username,password));
- }
如果参数名不能与这里的变量名保持一致,那么我们可以使用注解@RequestParam进行强制绑定,代码如下所示:
- @RequestMapping(method=RequestMethod.GET)
- publicvoidhello(@RequestParam("username")Stringu,
- @RequestParam("password")Stringp){
- System.out.println(accountService.verify(u,p));
- }
这比起我们之前写的代码有所简洁:
- @RequestMapping(method=RequestMethod.GET)
- publicvoidhello(HttpServletRequestrequest,HttpServletResponseresponse)
- throwsException{
- Stringusername=ServletRequestUtils.getRequiredStringParameter(
- request,"username");
- Stringpassword=ServletRequestUtils.getRequiredStringParameter(
- request,"password");
- System.out.println(accountService.verify(username,password));
- }
ServletRequestUtils类的工作已经由Spring底层实现了,我们只需要把参数名定义一致即可,其内部取参无需关心!
除了传入参数,我们还可以定义即将传出的参数,如加入ModelMap参数:
- @SuppressWarnings("unchecked")
- @RequestMapping(method=RequestMethod.GET)
- publicMaphello(Stringusername,Stringpassword,ModelMapmodel){
- System.out.println(accountService.verify(username,password));
- model.put("msg",username);
- returnmodel;
- }
这时,我们没有定义页面名称,Spring容器将根据请求名指定同名view,即如果是jap页面,则account.do->account.jsp!
不得不承认,这样写起来的确减少了代码量!
接着说输出参数!
通过ModelMap,我们可以绑定输出到的页面的参数,但最终我们将要返回到何种页面呢?再次查看AnnotationMethodHandlerAdapter源代码!
AnnotationMethodHandlerAdapter.java部分源代码——有关返回值部分:
- @SuppressWarnings("unchecked")
- publicModelAndViewgetModelAndView(MethodhandlerMethod,ClasshandlerType,ObjectreturnValue,
- ExtendedModelMapimplicitModel,ServletWebRequestwebRequest){
- if(returnValueinstanceofModelAndView){
- ModelAndViewmav=(ModelAndView)returnValue;
- mav.getModelMap().mergeAttributes(implicitModel);
- returnmav;
- }
- elseif(returnValueinstanceofModel){
- returnnewModelAndView().addAllObjects(implicitModel).addAllObjects(((Model)returnValue).asMap());
- }
- elseif(returnValueinstanceofMap){
- returnnewModelAndView().addAllObjects(implicitModel).addAllObjects((Map)returnValue);
- }
- elseif(returnValueinstanceofView){
- returnnewModelAndView((View)returnValue).addAllObjects(implicitModel);
- }
- elseif(returnValueinstanceofString){
- returnnewModelAndView((String)returnValue).addAllObjects(implicitModel);
- }
- elseif(returnValue==null){
- //Eitherreturnednullorwas'void'return.
- if(this.responseArgumentUsed||webRequest.isNotModified()){
- returnnull;
- }
- else{
- //Assumingviewnametranslation...
- returnnewModelAndView().addAllObjects(implicitModel);
- }
- }
- elseif(!BeanUtils.isSimpleProperty(returnValue.getClass())){
- //Assumeasinglemodelattribute...
- ModelAttributeattr=AnnotationUtils.findAnnotation(handlerMethod,ModelAttribute.class);
- StringattrName=(attr!=null?attr.value():"");
- ModelAndViewmav=newModelAndView().addAllObjects(implicitModel);
- if("".equals(attrName)){
- ClassresolvedType=GenericTypeResolver.resolveReturnType(handlerMethod,handlerType);
- attrName=Conventions.getVariableNameForReturnType(handlerMethod,resolvedType,returnValue);
- }
- returnmav.addObject(attrName,returnValue);
- }
- else{
- thrownewIllegalArgumentException("Invalidhandlermethodreturnvalue:"+returnValue);
- }
- }
返回值的定义十分庞大,或者说可怕的if-else多少有点让我觉得厌恶!
我们可以定义以下类型的返回值:
引用
ModelAndView
Model
View
Map
String
null
ModelAndView、Model和View都是Spring之前版本所特有的元素,Map对应于传入参数ModelMap,String定义页面名称,null即对应void类型方法!
最常用的实现方式如下:
- @SuppressWarnings("unchecked")
- @RequestMapping(method=RequestMethod.GET)
- publicStringhello(Stringusername,Stringpassword,ModelMapmodel){
- System.out.println(accountService.verify(username,password));
- model.put("msg",username);
- return"account";
- }
当然,对于我来说在返回值中写入这么一个字符串多少有点不能接受,于是我还是乐于使用输入参数ModelMap+输出参数Map的方式。
给出一个完整的AccountController实现:
AccountController.java
- /**
- *2010-1-23
- */
- packageorg.zlex.spring.controller;
- importjava.util.Map;
- importorg.springframework.beans.factory.annotation.Autowired;
- importorg.springframework.stereotype.Controller;
- importorg.springframework.ui.ModelMap;
- importorg.springframework.web.bind.annotation.RequestMapping;
- importorg.springframework.web.bind.annotation.RequestMethod;
- importorg.zlex.spring.service.AccountService;
- /**
- *
- *@author<ahref="mailto:zlex.dongliang@gmail.com">梁栋</a>
- *@version1.0
- *@since1.0
- */
- @Controller
- @RequestMapping("/account.do")
- publicclassAccountController{
- @Autowired
- privateAccountServiceaccountService;
- @SuppressWarnings("unchecked")
- @RequestMapping(method=RequestMethod.GET)
- publicMaphello(Stringusername,Stringpassword,ModelMapmodel){
- System.out.println(accountService.verify(username,password));
- model.put("msg",username);
- returnmodel;
- }
- }
最后说注解@Session
如果想将某个ModelMap中的参数指定到Session中,可以使用@Session注解,将其绑定为Session熟悉,代码如下所示:
- @Controller
- @RequestMapping("/account.do")
- @SessionAttributes("msg")
- publicclassAccountController{
- @Autowired
- privateAccountServiceaccountService;
- @SuppressWarnings("unchecked")
- @RequestMapping(method=RequestMethod.GET)
- publicMaphello(Stringusername,Stringpassword,ModelMapmodel){
- System.out.println(accountService.verify(username,password));
- model.put("msg",username);
- returnmodel;
- }
- }
当然,我们还需要配置一下对应的视图解析器,给出完整配置:
servelt.xml
- <?xmlversion="1.0"encoding="UTF-8"?>
- <beans
- xmlns="http://www.springframework.org/schema/beans"
- xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
- xmlns:p="http://www.springframework.org/schema/p"
- xmlns:context="http://www.springframework.org/schema/context"
- xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsd
- http://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsd">
- <context:component-scan
- base-package="org.zlex.spring.controller"/>
- <bean
- id="urlMapping"
- class="org.springframework.web.servlet.mvc.annotation.DefaultAnnotationHandlerMapping"/>
- <bean
- class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter"/>
- <bean
- id="jstlViewResolver"
- class="org.springframework.web.servlet.view.InternalResourceViewResolver"
- p:viewClass="org.springframework.web.servlet.view.JstlView"
- p:prefix="/WEB-INF/page/"
- p:suffix=".jsp"/>
- </beans>
这里使用了JstlView作为视图解析器。同时,指定前缀路径为"/WEB-INF/page/",后缀路径为".jsp"。也就是说,Spring容器将会在这个路径中寻找匹配的jsp文件!
注意加入xmlns:p="http://www.springframework.org/schema/p"命名空间!
再给出页面内容:
taglib.jsp
- <%@taglibprefix="c"uri="http://java.sun.com/jsp/jstl/core"%>
- <%@taglibprefix="fmt"uri="http://java.sun.com/jsp/jstl/fmt"%>
- <%@taglibprefix="sql"uri="http://java.sun.com/jsp/jstl/sql"%>
- <%@taglibprefix="x"uri="http://java.sun.com/jsp/jstl/xml"%>
- <%@taglibprefix="fn"uri="http://java.sun.com/jsp/jstl/functions"%>
- <%@taglibprefix="spring"uri="http://www.springframework.org/tags"%>
- <%@taglibprefix="form"uri="http://www.springframework.org/tags/form"%>
account.jap
- <%@pagelanguage="java"contentType="text/html;charset=UTF-8"
- pageEncoding="UTF-8"%>
- <%@includefile="/WEB-INF/page/taglib.jsp"%>
- <!DOCTYPEhtmlPUBLIC"-//W3C//DTDHTML4.01Transitional//EN""http://www.w3.org/TR/html4/loose.dtd">
- <html>
- <head>
- <metahttp-equiv="Content-Type"content="text/html;charset=UTF-8">
- <title>Account</title>
- </head>
- <body>
- <c:outvalue="${msg}"></c:out>
- </body>
- </html>
目录结构如图所示:
启动应用,最后将得到一个带有内容的页面,如图:
代码见附件!
相关推荐
在本篇《Spring注解学习手札(一)构建简单Web应用》中,我们将深入探讨如何使用Spring框架的注解来构建一个基本的Web应用程序。Spring框架是Java开发中的核心工具,尤其在企业级应用中广泛应用。它简化了依赖注入、...
【Spring注解学习手札】 在现代Java Web开发中,Spring框架因其强大的功能和灵活性而备受推崇。Spring注解的引入极大地简化了配置文件,提高了开发效率。本篇将聚焦于Spring MVC中的注解,通过构建一个简单的Web...
在本篇《Spring注解学习手札(四)持久层浅析》中,我们将深入探讨Spring框架在持久层的应用,特别是如何通过注解简化数据库操作。Spring作为一个强大的轻量级框架,提供了丰富的功能来处理数据访问,使得开发者可以...
这篇“Spring注解学习手札(五)——业务层事务处理”深入探讨了如何使用注解来管理应用程序中的事务,确保数据的一致性和完整性。Spring提供了声明式事务管理,使得开发者无需编写繁琐的事务控制代码,只需在方法上...
在本篇《Spring注解学习手札(三)表单页面处理》中,我们将深入探讨Spring框架中关于处理Web表单的关键注解和技术。在实际的Web开发中,表单处理是用户交互的重要组成部分,Spring提供了强大的支持,使得开发者能够...
在本篇《Spring注解学习手札(六)——测试》中,我们将深入探讨Spring框架中的测试支持,尤其是如何利用注解进行单元测试和集成测试。Spring为开发者提供了丰富的注解,使得测试代码更加简洁、易读且易于维护。本文...
在Spring框架中,注解是实现轻量级、声明式编程的重要工具,极大地简化了代码并提高了可维护性。本文将深入探讨`@ResponseBody`、`@RequestBody`和`@PathVariable`这三个关键注解,它们在处理HTTP请求和响应中的作用...
以上内容基于Snowolf的博客文章《Spring注解手札》系列,该系列文章详尽地介绍了Spring注解的使用,从构建简单的Web应用到控制层、表单处理、持久层以及事务管理和测试,覆盖了Spring注解的多个方面。通过这些实例,...
这份"Perl学习手札"是为那些想要深入理解Perl语言特性和应用的初学者或有一定经验的开发者准备的宝贵资源。 首先,Perl的核心特性在于其灵活性和表达力。Perl语法受到了C、sed、awk等多种语言的影响,这使得它在...
"Perl学习手札中文"是一份专为初学者设计的学习资料,旨在帮助读者快速掌握Perl语言的基础和高级特性。以下是对这些文件内容的概览: 1. **word.css**: 这个文件可能是样式表,用于定义文档中的排版和格式。在学习...
关于作者: 简信昌 “傲尔网”专案经理 博仲法律事务所资讯部门 台北Perl推广组 (Taipei.pm) 召集人 Newzilla召集人 目前专案: Open Source Foundry (OSSF) Newzilla 线上杂志 ...“Perl学习手札”
Perl学习手札.chmPerl学习手札.chmPerl学习手札.chm
- "spring-reference.pdf" 和 "Spring注解手札.pdf" 可能是详细的 Spring 参考文档和注解指南,对于深入学习 Spring 极为有用。 以上内容只是 Spring 框架的冰山一角,想要精通 Spring,还需要通过阅读文档、实践...
hibernate学习手札.z03
hibernate学习手札.z01
### Perl学习手札知识点概述 #### 1. 关于Perl - **1.1 Perl的历史**:Perl由Larry Wall在1987年创建,旨在为文本处理提供一种更强大的工具。随着时间的发展,Perl逐渐成为了脚本编程领域的领导者之一。 - **1.2 ...
通过深入学习“Perl学习手札”,你可以系统地掌握这些概念,并逐步成长为一个熟练的Perl程序员。记住,实践是检验知识的最好方式,所以不仅要理解理论,还要动手编写代码,解决实际问题。祝你在Perl的学习之旅中取得...
本资料完全来源于网上收集,对于作者并没有做过多的考证,考虑到作者本身,所以保持原作品不变只是略加整理,为用户提供一个方便的浏览方式。