SpringMVC主要有三个核心部分组成,DispatcherServlet、Controller、ViewResolver。
DispatcherServlet:
请求输入时:类似于一个带分配功能的Filter,其直接与前端交互,并截所有符合 url-pattern 的请求,并根据Mapping路径分发给处理对应请求的Controller。
请求处理完毕时:将ViewResolver渲染好的视图回传给前端。
Controller:
处理Http传来的请求,通常调用Service,再在Service中调用Dao持久层进行完整的数据处理,并将处理完毕的数据返回,返回以ModelAndView的形式,Model,通俗来讲,就是承载数据的一个HashMap,而View则是数据要发送的逻辑视图名,如果View缺省,默认是转发到HTTP发起的页面。
ViewResolver:
根据Controller处理好的数据,对指定目录下的文件进行渲染解析,完毕后将视图(不一定为页面、可能是Joson、Map各种数据类型,这根据Controller回传的数据决定)返回给DispatcherServlet。
DispatcherServlet配置
在web/web-inf/web.xml中配置
<?xml version="1.0" encoding="UTF-8"?> <web-app xmlns="http://xmlns.jcp.org/xml/ns/javaee" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://xmlns.jcp.org/xml/ns/javaee http://xmlns.jcp.org/xml/ns/javaee/web-app_3_1.xsd" version="3.1"> <!--配置contextLoaderListener以及文件地址,默认为web/web-inf/applicationContext.xml --> <context-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:applicationContext.xml</param-value> </context-param> <listener> <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener> <!--配置DispatcherServlet servlet-name可自取 --> <servlet> <servlet-name>dispatcher</servlet-name> <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class> <!-- 配置文件地址,默认为web/web-inf/ {servlet-name}-servlet.xml --> <init-param> <param-name>contextConfigLocation</param-name> <param-value>classpath:dispatcher-servlet.xml</param-value> </init-param> <!--自启动--> <load-on-startup>1</load-on-startup> </servlet> <!--配置拦截路径,此处为"/",可以改为其他 --> <servlet-mapping> <servlet-name>dispatcher</servlet-name> <url-pattern>/</url-pattern> </servlet-mapping> </web-app>
dispatcher-servlet.xml
配置开启注释、Json处理、视图解析器。
<?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" xmlns:mvc="http://www.springframework.org/schema/mvc" xmlns:beans="http://www.springframework.org/schema/beans" 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.xsd http://www.springframework.org/schema/mvc http://www.springframework.org/schema/mvc/spring-mvc.xsd"> <!--配合@ReponseBody使用,开启Jackson的转换--> <mvc:annotation-driven> <mvc:message-converters register-defaults="true"> <!--json处理--> <bean class="org.springframework.http.converter.json.MappingJackson2HttpMessageConverter"> <property name="supportedMediaTypes"> <list> <value>application/json;charset=UTF-8</value> <value>text/html;charset=UTF-8</value> </list> </property> </bean> </mvc:message-converters> </mvc:annotation-driven> <!--开启注解,base-package指向扫描的包地址,此处为Controller--> <context:component-scan base-package="Controller" use-default-filters="false"> <context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/> <context:include-filter type="annotation" expression="org.springframework.web.bind.annotation.ControllerAdvice"/> </context:component-scan> <!--Spring3.1开始的注解 HandlerMapping --> <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping"/> <!--Spring3.1开始的注解 HandlerAdapter --> <bean class="org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter"/> <!--配置视图解析器--> <bean class="org.springframework.web.servlet.view.InternalResourceViewResolver"> <property name="prefix" value="/" /> <property name="suffix" value=".jsp" /> </bean> </beans>
Controller
通过@Controller来定义一个Class为Controller。
通过@RequestMapping的Value值来定义访问路径,method来定义访问方式。
@Controller @RequestMapping(value="/login") public class LoginController { @RequestMapping(value = "/getDatas") public ModelAndView login( String username){ try { …… }catch (Exception e){ } } }
例如如上定义,想要访问login方法,如果在本地则为:localhost:8888/login/getDatas
Controller 返回值的类型多种多样,可以为map、List、String等,当然最通用的例子还是ModelAndView。
ModelAndView,其实际用途可以看成是Model 和 View两部分数据的返回,Model是数据部分,View是视图部分。数据最终传递到对应的视图上。
其更多是作为 处理一个校验,并完成转发的 用途,例如访问主页时需要检查用户权限,则可先访问Controller,在Controller校验完毕后,再重定向(当然也可以含参)到不同的逻辑页面(View)。
下面我们举一个含参的例子,跳转到succ.jsp或者error.jsp
@RequestMapping(value = "/getDatas") public ModelAndView login(){ ModelAndView modelAndView=new ModelAndView(); try { /*为modelAndView添加数据*/ modelAndView.addObject("TestData","1"); modelAndView.addObject("TestData1","2"); /*设置view的逻辑名称,SpringMVC会在视图解析器配置的目录下寻找该逻辑名称*/ modelAndView.setViewName("succ"); }catch (Exception e){ modelAndView.addObject("mesg","error"); modelAndView.setViewName("error"); } return modelAndView; }
我们可以通过再succ.jsp使用el表达式直接获取传递过来的值。
但其实很多时候我们接收到的都是AJAX传递过来的Json对象,完整解析后再给前端传出Json串,供前端使用JS进行渲染,这个时候我们的返回值就可以为Map、List、String类型的值。但是为了我们传出的值为Json,我们必须使用另一个标签@ResponseBody。
该注解用于将Controller的方法返回的对象,通过适当的HttpMessageConverter转换为指定格式后,写入到Response对象的body数据区。使用时机:返回的数据不是html标签的页面,而是其他某种格式的数据时(如json、xml等)使用;
我们将上面的方法加上@ResposeBody的标签,并使返回值为一个map
@ResponseBody @RequestMapping(value = "/getDatas") public Map login(){ Map<String,Object> map=new HashMap<>(); try { map.put("mesg","接口调用成功"); map.put("name","接口名称为login()"); }catch (Exception e){ map.put("mesg","error"); } return map; }
我们调用该接口可以看到其返回的使一个map的json串,并且也证明了在视图缺省的情况下,默认转发到请求的HTTP。
如果返回值为String,并添加了注释@ResponseBody,其返回的类型则为一个String,这一点需要注意(例:我们return "succ")
对于String返回类型还有一点特殊,即如果没有注释@ResponseBody,则SpringMVC默认其为一个逻辑视图名,以上面的 return "succ" 为例,我们将会跳转到SUCC页面
@RequestMapping(value = "/getDatas") public String login(){ String path=""; try { path="succ"; }catch (Exception e){ path="error"; } return path; }
我在视图解析器路径下建立了一个succ.jsp,并写了一个标题“SUCC PAGE”,可以已经完成了跳转。
这里的String就已经被SpringMVC默认为为View进行了命名的工作,类似于Struts2的Action。当然这样的跳转也可以像ModelAndView一样带参传递。
将Model置入Controller进行装载数据即可,见下:
@RequestMapping(value = "/getDatas") public String login(Model model){ String path=""; try { model.addAttribute("TestData","1"); model.addAttribute("TestData1","2"); path="succ"; }catch (Exception e){ path="error"; } return path; }
这样我们再次访问,就会发现两个EL表达式也取到了值。
String类型的返回值还有一个用于重定向的前缀"redirect:",当控制器方法返回的String值以“redirect:”开头的话,那么这个String不是用来查找视图的,而是用来知道浏览器进行重定向的路径。
例: return "redirect:succ.jsp"
前面我们讲述了一些Controller返回值类型的内容,现在我们来尝试为Controller传入一些参数。
Controller支持从表单直接拿值并自动匹配装配数据,这一点和Struts相同,例如表单两个输入一个username,一个password,直接post到Controller。
可见实体类以及字符串都可以拿到值。
更多的情况我们需要拿AJAX发送给我们的Json串,这个时候我们需要用到另一个注释@RequestBody来格式化取值,如下:
@RequestMapping(value = "/getDatas") public String login(@RequestBody user user,Model model){ //后方代码无用,打断点看值 String path=""; try { model.addAttribute("TestData","1"); model.addAttribute("TestData1","2"); path="login/edit"; }catch (Exception e){ path="error"; } return path; }
我们使用Soup UI直接进行端口测试,传送Json串
{ "username":"adim", "password":"12345" }
这样便可以拿到传进来的Json值了
注:@RequestBody接收的是一个Json对象的字符串,而不是一个Json对象。然而在ajax请求往往传的都是Json对象,后来发现用JSON.stringify(data)的方式就能将对象变成字符串。同时ajax请求的时候也要指定dataType:
"json",contentType:"application/json"这样就可以轻易的将一个对象传到Java端,使用@RequestBody即可绑定对象
另:
此处本人在开发过程中遇到了如果只有单个参数
如 public String login( String loginname) {}时,如果我传入
{“loginname”:"adim"}
Controller并取不到值,所以导致了单个参数也要封装的尴尬现象,还希望大佬们多多指点。
层面的注释
@Service
用于标注业务层组件
@Repository
用于标注数据访问组件,即DAO组件
@Controller
用于标注控制层组件(如struts中的action)
在Controller中调用ServiceImpl时,可直接使用@Autowire以及@Qualifier("xxx")在申明时进行注入。
其他常用的注释,来源:http://www.cnblogs.com/leskang/p/5445698.html
@ModelAttribute和 @SessionAttributes
代表的是:该Controller的所有方法在调用前,先执行此@ModelAttribute方法,可用于注解和方法参数中,可以把这个@ModelAttribute特性,应用在BaseController当中,所有的Controller继承BaseController,即可实现在调用Controller时,先执行@ModelAttribute方法。
@SessionAttributes即将值放到session作用域中,写在class上面。
具体示例参见下面:使用 @ModelAttribute 和 @SessionAttributes 传递和保存数据
@PathVariable
用于将请求URL中的模板变量映射到功能处理方法的参数上,即取出uri模板中的变量作为参数。如:
@Controller
public class TestController {
@RequestMapping(value="/user/{userId}/roles/{roleId}",method = RequestMethod.GET)
public String getLogin(@PathVariable("userId") String userId,
@PathVariable("roleId") String roleId){
System.out.println("User Id : " + userId);
System.out.println("Role Id : " + roleId);
return "hello";
}
@RequestMapping(value="/product/{productId}",method = RequestMethod.GET)
public String getProduct(@PathVariable("productId") String productId){
System.out.println("Product Id : " + productId);
return "hello";
}
@RequestMapping(value="/javabeat/{regexp1:[a-z-]+}",
method = RequestMethod.GET)
public String getRegExp(@PathVariable("regexp1") String regexp1){
System.out.println("URI Part 1 : " + regexp1);
return "hello";
}
}
@requestParam
@requestParam主要用于在SpringMVC后台控制层获取参数,类似一种是request.getParameter("name"),它有三个常用参数:defaultValue = "0", required = false, value = "isApp";defaultValue 表示设置默认值,required 铜过boolean设置是否是必须要传入的参数,value 值表示接受的传入的参数类型。