目录
Spring4.1新特性——页面自动化测试框架Spring MVC Test HtmlUnit简介
Spring 4.1对Spring MVC部分做的增强是最多的,提供了一些视图解析器的mvc标签实现简化配置、提供了GroovyWebApplicationContext用于 Groovy web集成、提供了Gson、protobuf的HttpMessageConverter、提供了对groovy-templates模板的支持、 JSONP的支持、对Jackson的@JsonView的支持等。
1、GroovyWebApplicationContext
在Spring 4.1之前没有提供Web集成的ApplicationContext,在《Spring4新特性——Groovy Bean定义DSL》中我们自己去实现的com.sishuok.spring4.context.support.WebGenricGroovyApplicationContext,而4.1其已经提供了相应实现,直接把《Spring4新特性——Groovy Bean定义DSL》配置中的相应类改掉即可。
2、视图解析器标签
之前我们都是这样定义视图解析器:
- <bean id="mvcVelocityEngine" class="org.springframework.web.servlet.view.velocity.VelocityConfigurer">
- <property name="resourceLoaderPath" value="/WEB-INF/vm/,classpath:com/github/zhangkaitao" />
- </bean>
- <bean id="viewResolver" class="org.springframework.web.servlet.view.velocity.VelocityViewResolver">
- <property name="prefix" value=""/>
- <property name="suffix" value=".vm"/>
- <property name="cache" value="false"/>
- </bean>
而现在我们可以使用MVC标签定义:
- <mvc:velocity-configurer resource-loader-path="/WEB-INF/vm/,classpath:com/github/zhangkaitao"/>
- <mvc:view-resolvers>
- <mvc:velocity cache-views="false" prefix="" suffix=".vm"/>
- </mvc:view-resolvers>
再来看一个更复杂的例子:
- <mvc:velocity-configurer resource-loader-path="/WEB-INF/vm/,classpath:com/github/zhangkaitao"/>
- <mvc:groovy-configurer resource-loader-path="classpath:templates/" cache-templates="false"/>
- <mvc:view-resolvers>
- <mvc:content-negotiation>
- <mvc:default-views>
- <bean class="org.springframework.web.servlet.view.json.MappingJackson2JsonView">
- <property name="jsonpParameterNames">
- <set>
- <value>jsonp</value>
- <value>callback</value>
- </set>
- </property>
- </bean>
- </mvc:default-views>
- </mvc:content-negotiation>
- <mvc:velocity cache-views="false" prefix="" suffix=".vm"/>
- <mvc:groovy cache-views="false" suffix=".tpl"/>
- </mvc:view-resolvers>
mvc:content-negotiation用于定义内容协商的视图解析器,且内部可以定义默认视图;然后我们又定义了mvc:velocity和mvc:groovy两个视图解析器;它们会按照顺序进行解析。另外几个视图解析器是:
mvc:freemarker
mvc:bean-name
mvc:jsp
这种方式有一个很大的问题就是只能做默认配置,如果想自定义其属性值就搞不定了,估计当时开发的人考虑不全或没有经验。
3、控制器标签
Spring 4.1提供了更丰富的控制器标签:
3.1、重定向视图控制器标签
- <mvc:redirect-view-controller
- path="/redirect"
- redirect-url="/status"
- context-relative="true"
- status-code="301"
- keep-query-params="true"/>
3.2、状态控制器标签
- <mvc:status-controller path="/status" status-code="200"/>
3.3、带状态的视图控制器标签
- <mvc:view-controller path="/error/**" status-code="200"/>
4、Groovy Template引擎集成
Spring 4.1提供了对Groovy Template模板引擎的集成,其是一种DSL风格的模板引擎,其也是最早在Spring Boot中引入的。
4.1、Spring配置文件
- <mvc:groovy-configurer resource-loader-path="classpath:templates/" cache-templates="false"/>
- <mvc:view-resolvers>
- <mvc:groovy cache-views="false" suffix=".tpl"/>
- </mvc:view-resolvers>
4.2、模板heelo.tpl
- yieldUnescaped '<!DOCTYPE html>'
- html {
- head {
- title('hello groovy templates')
- }
- body {
- div("hello $user.name")
- }
- }
具体语法请参考官方文档。
5、 Jackson @JsonView支持
可以使用@JsonView来分组渲染JSON数据,按需展示JSON数据。
5.1、模型
- public class User implements Serializable {
- public static interface OnlyIdView {}
- public static interface OnlyNameView {}
- public static interface AllView extends OnlyIdView, OnlyNameView {}
- @JsonView(OnlyIdView.class)
- private Long id;
- @JsonView(OnlyNameView.class)
- private String name;
- ……
- }
定义了三个视图:OnlyIdView、OnlyNameView和AllView。
5.2、控制器
- @RestController
- public class JacksonJsonViewController {
- @RequestMapping("/jackson1")
- @JsonView(User.OnlyIdView.class)
- public User test1() {
- return new User(1L, "zhangsan");
- }
- @RequestMapping("/jackson2")
- @JsonView(User.OnlyNameView.class)
- public User test2() {
- return new User(1L, "zhangsan");
- }
- @RequestMapping("/jackson3")
- @JsonView(User.AllView.class) //可以省略
- public User test3() {
- return new User(1L, "zhangsan");
- }
- }
使用@JsonView控制渲染哪些数据。
6、Jsonp支持
6.1、MappingJackson2JsonView提供的支持
- <bean class="org.springframework.web.servlet.view.json.MappingJackson2JsonView">
- <property name="jsonpParameterNames">
- <set>
- <value>jsonp</value>
- <value>callback</value>
- </set>
- </property>
- </bean>
然后访问如http://localhost:8080/json?callback=callback即可得到JSONP响应:callback({"user":{"id":1,"name":"zhangsan"}});。
6.2、对使用HttpMessageConverter的@ResponseBody的支持
- @Order(2)
- @ControllerAdvice(basePackages = "com.github")
- public class JsonpAdvice extends AbstractJsonpResponseBodyAdvice {
- public JsonpAdvice() {
- super("callback", "jsonp"); //指定jsonpParameterNames
- }
- }
访问http://localhost:8080/jackson1?callback=callback即可看到JSONP响应。
@ContollerAdvice的作用请参考《Spring3.2新注解@ControllerAdvice》,basePackages用于指定对哪些包里的Controller起作用。
6.3、ResponseBodyAdvice
我们之前实现的JsonpAdvice其继承自AbstractJsonpResponseBodyAdvice,而AbstractJsonpResponseBodyAdvice继承自ResponseBodyAdvice,其作用是在响应体写出之前做一些处理:
- @Order(1)
- @ControllerAdvice(basePackages = "com.github")
- public class MyResponseBodyAdvice implements ResponseBodyAdvice<Object> {
- @Override
- public boolean supports(MethodParameter methodParameter, Class<? extends HttpMessageConverter<?>> converterType) {
- return methodParameter.getMethod().getReturnType().isAssignableFrom(User.class);
- }
- @Override
- public Object beforeBodyWrite(
- Object obj, MethodParameter methodParameter, MediaType mediaType,
- Class<? extends HttpMessageConverter<?>> converterType,
- ServerHttpRequest serverHttpRequest, ServerHttpResponse serverHttpResponse) {
- User user = ((User)obj);
- user.setName("---" + user.getName() + "---");
- return user;
- }
- }
1、supports指定支持哪些类型的方法进行处理,此处是返回值为User的;2、我们得到User对象然后在名字前后拼上”---“ ;3、可以指定多个ResponseBodyAdvice,使用@Order指定顺序。访问http://localhost:8080 /jackson2?callback=callback可以看到效果。
7、Gson HttpMessageConverter
7.1、Spring配置
- <mvc:annotation-driven>
- <mvc:message-converters>
- <bean class="org.springframework.http.converter.json.GsonHttpMessageConverter"/>
- </mvc:message-converters>
- </mvc:annotation-driven>
使用方式和Jackson Json类似。本文使用的是<gson.version>2.2.4</gson.version>版本。
8、Protobuf HttpMessageConverter
8.1、Spring配置
- <mvc:annotation-driven>
- <mvc:message-converters>
- <bean class="org.springframework.http.converter.protobuf.ProtobufHttpMessageConverter">
- <constructor-arg>
- <bean class="com.github.zhangkaitao.web.controller.MyExtensionRegistryInitializer"/>
- </constructor-arg>
- </bean>
- </mvc:message-converters>
- </mvc:annotation-driven>
8.2、定义protobuf message(proto/user.proto)
- package com.github.zhangkaitao.pb;
- option java_package = "com.github.zhangkaitao.pb";
- option java_outer_classname = "UserProtos";
- message User {
- optional int64 id = 1;
- optional string name = 2;
- }
8.3、添加maven插件自动把protobuf message转化成Java代码
- <plugin>
- <groupId>com.google.protobuf.tools</groupId>
- <artifactId>maven-protoc-plugin</artifactId>
- <version>0.1.10</version>
- <executions>
- <execution>
- <id>generate-sources</id>
- <goals>
- <goal>compile</goal>
- </goals>
- <phase>generate-sources</phase>
- <configuration>
- <protoSourceRoot>${basedir}/src/main/proto/</protoSourceRoot>
- <includes>
- <param>**/*.proto</param>
- </includes>
- </configuration>
- </execution>
- </executions>
- <configuration>
- <protocExecutable>D:/software/protoc.exe</protocExecutable>
- </configuration>
- </plugin>
8.4、测试控制器
- @RestController
- public class ProtobufController {
- @RequestMapping("/proto/read")
- public ResponseEntity<UserProtos.User> protoRead() {
- return ResponseEntity.ok(UserProtos.User.newBuilder().setId(1).setName("zhangsan").build());
- }
- @RequestMapping("/proto/write")
- public ResponseEntity<UserProtos.User> protoRead(RequestEntity<UserProtos.User> requestEntity) {
- System.out.println("server===\n" + requestEntity.getBody());
- return ResponseEntity.ok(requestEntity.getBody());
- }
- }
8.5、测试用例(com.github.zhangkaitao.proto.ProtoTest)
- @Test
- public void testRead() {
- HttpHeaders headers = new HttpHeaders();
- RequestEntity<UserProtos.User> requestEntity =
- new RequestEntity<UserProtos.User>(headers, HttpMethod.POST, URI.create(baseUri + "/proto/read"));
- ResponseEntity<UserProtos.User> responseEntity =
- restTemplate.exchange(requestEntity, UserProtos.User.class);
- System.out.println(responseEntity.getBody());
- }
- @Test
- public void testWrite() {
- UserProtos.User user = UserProtos.User.newBuilder().setId(1).setName("zhangsan").build();
- HttpHeaders headers = new HttpHeaders();
- RequestEntity<UserProtos.User> requestEntity =
- new RequestEntity<UserProtos.User>(user, headers, HttpMethod.POST, URI.create(baseUri + "/proto/write"));
- ResponseEntity<UserProtos.User> responseEntity =
- restTemplate.exchange(requestEntity, UserProtos.User.class);
- System.out.println(responseEntity.getBody());
- }
测试用例知识请参考《Spring MVC测试框架详解——服务端测试》和《Spring MVC测试框架详解——客户端测试》。
测试过程中会抛出:
- Caused by: java.lang.UnsupportedOperationException
- at java.util.Collections$UnmodifiableMap.put(Collections.java:1342)
- at org.springframework.http.HttpHeaders.set(HttpHeaders.java:869)
- at org.springframework.http.converter.protobuf.ProtobufHttpMessageConverter.setProtoHeader(ProtobufHttpMessageConverter.java:196)
这是因为ProtobufHttpMessageConverter会修改响应头,但是ResponseEntity构造时HttpHeaders是不允许修改的。暂时解决办法是注释掉:
- //setProtoHeader(outputMessage, message);
9、RequestEntity/ResponseEntity
Spring 4.1提供了ResponseEntity配对的RequestEntity,使用方式和HttpEntity一样。具体可以参考 com.github.zhangkaitao.web.controller.RequestResponseEntityController。
10、MvcUriComponentsBuilder
其作用可以参考《Spring4新特性——注解、脚本、任务、MVC等其他特性改进》,Spring 4.1又提供了一个新的方法MvcUriComponentsBuilder.fromMappingName用于根据控制器方法来生成请求URI。
- @RestController
- public class MvcUriComponentsBuilderController {
- @RequestMapping("/uri")
- public String mvcUriComponentsBuilder1() {
- return MvcUriComponentsBuilder.fromMappingName("MUCBC#mvcUriComponentsBuilder1").build();
- }
- @RequestMapping("/uri/{id}")
- public String mvcUriComponentsBuilder2(@PathVariable Long id) {
- return MvcUriComponentsBuilder.fromMappingName("MUCBC#mvcUriComponentsBuilder2").arg(0, "123").build();
- }
- }
规则是“控制器所有大写字母#方法名”找到相应的方法。 另外可以直接在页面中使用如下方式获取相应的URI:
- ${s:mvcUrl('MUCBC#mvcUriComponentsBuilder2').arg(0,"123").build()}
如上方式只能在正常EL 3.0的容器中运行,可参考《Expression Language 3.0新特性》。
11、MockRestServiceServer
MockRestServiceServer目前提供了对AsyncRestTemplate的支持,使用方式和RestTemplate一样。可参考《Spring MVC测试框架详解——客户端测试》。
12、MockMvcConfigurer
Spring 4.1提供了MockMvcConfigurer用于进行一些通用配置,使用方式如下:
- mockMvc = MockMvcBuilders.webAppContextSetup(context).apply(defaultSetup()).build();
MockMvcConfigurer实现:
- private MockMvcConfigurer defaultSetup() {
- return new MockMvcConfigurer() {
- @Override
- public void afterConfigurerAdded(ConfigurableMockMvcBuilder<?> configurableMockMvcBuilder) {
- configurableMockMvcBuilder.alwaysExpect(status().isOk());
- }
- @Override
- public RequestPostProcessor beforeMockMvcCreated(ConfigurableMockMvcBuilder<?> configurableMockMvcBuilder, WebApplicationContext webApplicationContext) {
- return new RequestPostProcessor() {
- @Override
- public MockHttpServletRequest postProcessRequest(MockHttpServletRequest mockHttpServletRequest) {
- mockHttpServletRequest.setAttribute("aa", "aa");
- return mockHttpServletRequest;
- }
- };
- }
- };
- }
可以在如上实现中进行一些通用配置,如安全(往Request中扔安全对象之类的)。测试用例可参考com.github.zhangkaitao.proto.ProtoTest2。
相关文章
https://spring.io/blog/2014/05/28/using-the-innovative-groovy-template-engine-in-spring-boot
Spring4新特性——注解、脚本、任务、MVC等其他特性改进
Spring4新特性
Spring4新特性——集成Bean Validation 1.1(JSR-349)到SpringMVC
Spring4新特性——注解、脚本、任务、MVC等其他特性改进
源码下载
https://github.com/zhangkaitao/spring4-1-showcase/tree/master/spring4.1-groovy
https://github.com/zhangkaitao/spring4-1-showcase/tree/master/spring4.1-mvc
评论
kaitao能具体说下这段是什么意思么?
比如下面的红色部分
<bean id="viewResolver" class="org.springframework.web.servlet.view.velocity.VelocityViewResolver">
<property name="prefix" value=""/>
<property name="suffix" value=".vm"/>
<property name="cache" value="false"/>
<property name="contentType" value="text/html;charset=${encoding}"/>
<property name="exposeRequestAttributes" value="true"/>
<property name="requestContextAttribute" value="request"/>
</bean>
谢谢, 原来是这个意思。 按你的意思是可以设计成和p命名空间类似的实现吗?
可以的
kaitao能具体说下这段是什么意思么?
比如下面的红色部分
<bean id="viewResolver" class="org.springframework.web.servlet.view.velocity.VelocityViewResolver">
<property name="prefix" value=""/>
<property name="suffix" value=".vm"/>
<property name="cache" value="false"/>
<property name="contentType" value="text/html;charset=${encoding}"/>
<property name="exposeRequestAttributes" value="true"/>
<property name="requestContextAttribute" value="request"/>
</bean>
谢谢, 原来是这个意思。 按你的意思是可以设计成和p命名空间类似的实现吗?
kaitao能具体说下这段是什么意思么?
比如下面的红色部分
<bean id="viewResolver" class="org.springframework.web.servlet.view.velocity.VelocityViewResolver">
<property name="prefix" value=""/>
<property name="suffix" value=".vm"/>
<property name="cache" value="false"/>
<property name="contentType" value="text/html;charset=${encoding}"/>
<property name="exposeRequestAttributes" value="true"/>
<property name="requestContextAttribute" value="request"/>
</bean>
kaitao能具体说下这段是什么意思么?
相关推荐
Spring4.1版本是该框架的一个重要里程碑,它引入了诸多改进和新特性,旨在提高开发效率和应用性能。在这个"Spring4.1官方jar包完整版"中,我们可以找到所有必要的组件,包括核心容器、数据访问/集成、Web、AOP(面向...
Spring 4.1开始全面支持Java 8的新特性,如Lambda表达式、日期和时间API等,让开发者能够充分利用Java 8的优势。 8. **Spring Boot** 虽然Spring Boot不是Spring 4.1的一部分,但4.1版本与Spring Boot的结合使得...
对于数据库访问,Spring 4.1加强了JPA、Hibernate和MyBatis的支持,提供了事务管理和数据验证的功能。此外,还支持NoSQL数据库如MongoDB和Cassandra。 8. **Spring WebSocket** Spring 4.1引入了对WebSocket协议...
总的来说,这个"java spring4.1.x源码"压缩包是学习和研究Spring框架的宝贵资料,无论是对于初学者还是经验丰富的开发者,都能从中受益匪浅。通过深入研究源码,可以提升对Spring的理解,更好地利用Spring来构建高效...
7. **Web MVC增强**: MVC框架在4.1中进行了优化,包括性能提升、错误处理改进以及模板引擎的集成。例如,`ModelAndView`现在支持`ModelMap`,使得模型数据处理更加灵活。 8. **RESTful服务**: Spring 4.1提供了更好...
9. **RESTful支持**:Spring 4.1进一步加强了对RESTful风格的Web服务支持,可以通过`@RestController`注解快速创建API接口。 10. **模板引擎**:Spring 4.1对FreeMarker和Thymeleaf等模板引擎的支持更佳,使得...
6. **Integration with other technologies**:Spring 4.1还加强了与其他技术的集成,如RESTful服务的支持、JSON序列化和反序列化以及对Java 8新特性的支持,如日期时间API,使得开发者可以更好地利用现有技术和新的...
在Web层,Spring MVC在4.1版本中引入了更好的RESTful支持,如`@JsonView`用于控制JSON序列化的视图,以及`@ControllerAdvice`用于全局异常处理。另外,`ModelAndView`对象现在支持直接返回`String`,简化了视图的...
Spring4.1新特性示例 Spring核心部分及其他 Spring缓存框架增强 初步引发和事件机制的异常处理 数据库集成测试脚本初始化 Spring MVC增强 页面自动化测试框架Spring MVC Test HtmlUnit简介 静态资源处理增强
在本项目中,我们主要关注的是一个基于Spring MVC 4.1、Spring 4.1、MyBatis 3.2以及Spring Security 3.2的Maven环境的搭建过程。下面将详细介绍这些技术栈的核心概念、它们如何协同工作以及如何在实际开发环境中...
这意味着你可以利用JDK 1.7的一些特性,如try-with-resources语句、多线程改进等,同时享受到Spring 4.1提供的新特性和优化。 5. **jar包和实例**: 压缩包中的jar包很可能包含了Spring 4.1的核心库和其他依赖,...
Spring4.1版本是Spring框架的一个重要里程碑,引入了许多新特性和改进,提高了性能和易用性。 1. **依赖注入(DI)**:Spring的核心特性之一,允许开发者在运行时将对象及其依赖关系进行管理,而不是硬编码到代码中。...
在4.1版本中,Spring引入了Java配置的增强,使得无需XML配置就能轻松地搭建应用,同时支持数据访问对象(DAO)的简化,与各种持久层框架如Hibernate的集成更加紧密。 Struts 2作为MVC(模型-视图-控制器)架构的一...
1. **Spring 4.1的关键特性**: - **Java 8支持**:Spring 4.1全面支持Java 8,包括Lambda表达式和日期/时间API的集成。 - **配置增强**:引入了`@Profile`注解,允许根据特定环境加载不同的配置类。同时,`@Bean`...
Spring MVC 4.1引入了一些新特性,如支持JSR 353 JSON绑定,改进了异常处理,以及更好的类型安全的模型属性绑定等。 2. **Spring** Spring 框架是Java企业级应用的核心,它提供了一种依赖注入(DI)和面向切面编程...
在使用 Spring 和 Spring MVC 4.1 的过程中,你需要以下的 jar 包: 1. **spring-core.jar**:包含 Spring 框架的基本组件,如 Bean 工厂、依赖注入等。 2. **spring-context.jar**:提供了上下文环境、事件、国际化...
Spring4.1、Hibernate4.3.7和Struts2.3.6是这三个框架的特定版本,它们各自引入了新的特性和优化,同时保持着对前一版本的兼容性。下面将详细介绍这三个框架及其整合案例中的关键知识点。 **Spring4.1** Spring是一...
spring-webmvc-4.1.0.RELEASE.jar用于Spring框架的mvc注解
### Spring Framework 4.1.x Web MVC框架知识点详解 #### 一、Spring Web MVC框架概述 **Spring Web MVC**框架是Spring框架中的一个重要组成部分,它主要用于处理基于HTTP协议的请求,构建灵活且易于扩展的Web应用...
Hibernate4.1在性能和功能上都有所提升,包括支持JPA 2.1规范、优化的缓存机制以及对新数据库特性的支持。 在这个SSH整合中,Struts2负责处理HTTP请求和响应,Spring管理整个应用的bean,包括Struts2的Action和...