第14章 JSON数据交互和RESTful支持
学习目标
● 了解JSON的数据结构
● 掌握Spring MVC中JSON数据交互的使用
● 熟悉RESTful风格的请求样式
● 掌握Spring MVC中RESTful风格请求的使用
14.1 JSON数据交互
它与XML非常相似,都是用于存储数据的;但JSON相对于XML来说,解析速度更快,占用空间更小
14.1.1 JSON概述
JSON(JavaScript Object Notation, JS对象标记)是一种轻量级的数据交换格式。它是基于JavaScript的一个子集,使用了C、C++、C#、Java、JavaScript、Perl、Python等其他语言的约定,采用完全独立于编程语言的文本格式来存储和表示数据。这些特性使JSON成为理想的数据交互语言,它易于阅读和编写,同时也易于机器解析和生成。
与XML一样,JSON也是基于纯文本的数据格式。初学者可以使用JSON传输一个简单的String、Number、Boolean,也可以传输一个数组或者一个复杂的Object对象。
1.对象结构
对象结构以“{”开始,以“}”结束。中间部分由0个或多个以英文“, ”分隔的name/value对构成(注意name和value之间以英文“:”分隔),
{
key1:value1,
key2:value2,
...
}
(key)必须为String类型,值(value)可以是String、Number、Object、Array等数据类型
例子
{"city":"Beijing", "street":"Xisanqi", "postcode":100096}
2.数组结构
数组结构以“[”开始,以“]”结束。中间部分由0个或多个以英文“, ”分隔的值的列表组成
[
value1,
value2,
...
]
例子
["abc",12345, false, null]
(对象、数组)数据结构也可以分别组合构成更为复杂的数据结构
{
"name": "zhangsan"
"hobby":["篮球", "羽毛球", "游泳"]
"address":{
"city":"Beijing"
"street":"Xisanqi"
"postcode":100096
}
}
如果使用JSON存储单个数据(如“abc”),一定要使用数组的形式,不要使用Object形式,因为Object形式必须是“名称:值”的形式
14.1.2 JSON数据转换
HttpMessageConverter<T>接口实现浏览器与控制器类(Controller)之间的数据交互,该接口主要用于将请求信息中的数据转换为一个类型为T的对象,并将类型为T的对象绑定到请求方法的参数中,或者将对象转换为响应信息传递给浏览器显示。
实现类:默认MappingJackson2HttpMessageConverter
利用Jackson开源包读写JSON数据,将Java对象转换为JSON对象和XML文档,同时也可以将JSON对象和XML文档转换为Java对象。
Jackson的开源包:
· jackson-annoations-2.8.8.jar:JSON转换注解包。
· jackson-core-2.8.8.jar:JSON转换核心包。
· jackson-databind-2.8.8.jar:JSON转换的数据绑定包。
两个重要的JSON格式转换注解,分别为@RequestBody和@ResponseBody
1.导包
2.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_4_0.xsd"
version="4.0">
<display-name>chapeter14</display-name>
<welcome-file-list>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
<servlet>
<servlet-name>springmvc</servlet-name>
<servlet-class>
org.springframework.web.servlet.DispatcherServlet
</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc-config.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>springmvc</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
</web-app>
3.springmvc-config.xml
<?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"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans-4.3.xsd
http://www.springframework.org/schema/mvc
http://www.springframework.org/schema/mvc/spring-mvc-4.3.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context-4.3.xsd
http://www.springframework.org/schema/cache
http://www.springframework.org/schema/cache/spring-cache.xsd">
<!-- 指定需要扫描的包-->
<context:component-scan base-package="com.itheima.controller"/>
<!-- 配置注解驱动-->
<mvc:annotation-driven />
<!-- 配置静态资源的访问映射,此配置中的文件,将不被前端拦截器拦截-->
<mvc:resources mapping="/js/**" location="/js/"/>
<!-- 定义视图解析器 -->
<bean id="viewResolver" class= "org.springframework.web.servlet.view.InternalResourceViewResolver">
<!-- 设置前缀 -->
<property name="prefix" value="/WEB-INF/jsp/" />
<!-- 设置后缀 -->
<property name="suffix" value=".jsp" />
</bean>
</beans>
组件扫描器、视图解析器、Spring MVC的注解驱动<mvc:annotation-drivern />、静态资源访问映射<mvc:resources …/>
<mvc:annotation-drivern/>配置会自动注册RequestMappingHandlerMapping和RequestMappingHandlerAdapter两个Bean,并提供对读写XML和读写JSON等功能的支持
增加了静态资源的访问映射配置后,程序会自动地去配置路径下找静态的内容。
<mvc:resources>标签属性及说明
4.pojo类
User类属性username和password,get,set,toString方法
5.index.jsp
<head>
<title>测试JSON交互</title>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
</head>
<script type="text/javascript" src="${pageContext.request.contextPath }/js/jquery.min.js"></script>
<script type="text/javascript">
function testJson(){
var username= $("#username").val();
var password= $("#password").val();
$.ajax({
url :"${pageContext.request.contextPath }/testJson",
type :"post",
data :JSON.stringify({username:username,password:password}),
// 定义发送请求的数据格式为JSON字符串
contentType :"application/json; charset=UTF-8",
// 定义回调响应的数据格式为JSON字符串,该属性可省略
dataType :"json",
// 成功响应的结果
success :function (data){
if(data != null){
alert("您输入的用户名:"+data.username+"密码为:"+data.password);
}
}
});
}
</script>
<body>
<form>
用户名:<input type="text" name="username" id="username" /><br/>
密 码:<input type="password" name="password" id="password"/><br/>
<input type="button" value="测试JSON交互" onclick="testJson()" />
</form>
</body>
在函数中使用了jQuery的AJAX方式将JSON格式的用户名和密码传递到以“/testJson”结尾的请求中。
· data:即请求时携带的数据,当使用JSON格式时,要注意编写规范。· contentType:当请求数据为JSON格式时,值必须为application/json。
· data:即请求时携带的数据,当使用JSON格式时,要注意编写规范。· contentType:当请求数据为JSON格式时,值必须为application/json。
· dataType:当响应数据为JSON时,可以定义dataType属性,并且值必须为json。其中dataType : "json"也可以省略不写,页面会自动识别响应的数据格式。
6.UserController
@RequestMapping("/testJson")
@ResponseBody
public User testJson(@RequestBody User user){
// 打印接收的JSON格式数据
System.out.println(user);
return user;
}
使用注解方式定义了一个控制器类,并编写了接收和响应JSON格式数据的testJson()方法,在方法中接收并打印了接收到的JSON格式的用户数据,然后返回了JSON格式的用户对象。
7.启动服务器
问题:Failed to load resource: the server responded with a status of 404 (Not Found) :8080/favicon.ico:1
解决方法:检查springmvc-config.xml文件静态资源访问映射配置
<mvc:resources mapping="/js/**" location="/js/"/>
问题:Failed to load resource: the server responded with a status of 404 (Not Found) jquery.min.js:1
解决方法:把js文件挪出WEB-INF(非rest风格代码下js资源通常为静态资源,放在web-inf下会加载不上)
问题:Failed to load resource: the server responded with a status of 404 (Not Found):8080/pro07/textJson:1
解决方法:url里拼错了test……
问题:Failed to load resource: the server responded with a status of 415 (Unsupported Media Type) :8080/pro07/testJson:1
解决方法:在项目结构中手动加入jakson依赖
问题:Failed to load resource: the server responded with a status of 400 (Bad Request)
问题描述POST The request sent by the client was syntactically incorrect.
原因:请求数据和controller处理方法中的参数不匹配,pojo类User类的password写成了Integer类型,
解决方法:User类的password改为String类型
1.JSON转换器的配置
(1).<mvc:annotation-drivern />
(2).<bean>标签
<! -- <bean>标签配置注解方式的处理器映射器和处理器适配器必须配对使用-->
<! -- 使用<bean>标签配置注解方式的处理器映射器 -->
<bean class="org.springframework.web.servlet.mvc.method .annotation.RequestMappingHandlerMapping" /> <! -- 使用<bean>标签配置注解方式的处理器适配器 -->
<bean class="org.springframework.web.servlet.mvc.method .annotation.RequestMappingHandlerAdapter"> <property name="messageConverters">
<list>
<! -- 在注解适配器中配置JSON转换器 -->
<bean class="org.springframework.http.converter.json .MappingJackson2HttpMessageConverter"/>
</list>
</property>
</bean>
使用<bean>标签配置方式配置JSON转换器时,需要同时配置处理器映射器和处理器适配器,并且JSON转换器是配置在适配器中
2.配置静态资源访问的方式
(1)<mvc:resources>
(2)<mvc:default-servlet-handler>
<mvc:default-servlet-handler>
会在Spring MVC上下文中定义一个org.springframework.web.servlet.resource.DefaultServletHttpRequestHandler(即默认的Servlet请求处理器)。它会像一个检查员,对进入DispatcherServlet的URL进行筛查。如果发现是静态资源的请求,就将该请求转由Web服务器默认的Servlet处理,默认的Servlet就会对这些静态资源放行;如果不是静态资源的请求,才由DispatcherServlet继续处理。
一般Web服务器默认的Servlet名称是"default",因此DefaultServletHttpRequestHandler可以找到它。如果使用的Web应用服务器默认的Servlet名称不是"default",则需要通过default-servlet-name属性显示指定,具体方式如下。
<mvc:default-servlet-handler default-servlet-name="Servlet名称" />
而Web服务器的Servlet名称是由使用的服务器确定的,常用服务器及其Servlet名称如下。
· Tomcat、Jetty、JBoss和and GlassFish默认Servlet的名称——"default"。
· Google App Engine默认Servlet的名称——"_ah_default"。
· Resin默认Servlet的名称——"resin-file"。
· WebLogic默认Servlet的名称——"FileServlet"。
· WebSphere默认Servlet的名称——"SimpleFileServlet"。
(3)激活Tomcat默认的Servlet来处理静态文件访问
web.xml中
<! --激活tomcat的静态资源拦截,需要哪些静态文件,再往下追加-->
<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>*.js</url-pattern>
</servlet-mapping>
<servlet-mapping>
<servlet-name>default</servlet-name>
<url-pattern>*.css</url-pattern>
</servlet-mapping>
...
配置了<servlet-mapping>元素来激活Tomcat默认的Servlet来处理静态文件,我们还可以根据需要继续追加<servlet-mapping>。此种配置方式和上一种方式本质上是一样的,都是使用Web服务器默认的Servlet来处理静态资源文件的访问。其中Servelt名称(即<servlet-name>元素的值)也是由使用的服务器来确定的,不同的服务器需要使用不同的名称。
优缺点:
· 第一和第三种配置方式可以选择性的释放静态资源。
· 第二种配置方式配置相对简单,只需要一行代码,就可以释放所有静态资源。
· 第二和第三种配置方式会导致项目移植性较差,需要根据具体的Web服务器来更改Servlet名称。
· 第三种配置方式运行效率更高,因为服务器启动时已经加载了web.xml中的静态资源。
14.2 RESTful支持
14.2.1 什么是RESTful
RESTful也称之为REST(Representational State Transfer),可以将它理解为一种软件架构风格或设计风格,而不是一个标准。简单来说,RESTful风格就是把请求参数变成请求路径的一种风格。
传统:
http://.../queryItems? id=1
RESTful风格:
http://.../items/1
RESTful风格中的URL不存在动词形式。
在HTTP请求中,使用put、delete、post和get方式分别对应添加、删除、修改和查询的操作。
14.2.2 应用案例——用户信息查询
1.UserController
@RequestMapping(value = "/user/{id}",method = RequestMethod.GET)
@ResponseBody
public User selectUser(@PathVariable("id") String id){
// 查看数据接收
System.out.println("id = " + id);
User user = new User();
// 模拟根据id查出到用户对象数据
if(id.equals("123456")){
user.setUsername("shu");
}
// 返回JSON格式的数据
return user;
}
方法中的@PathVariable("id")注解则用于接收并绑定请求参数,它可以将请求URL中的变量映射到方法的形参上,如果请求路径为“/user/{id}”,即请求参数中的id和方法形参名称id一样,则@PathVariable后面的“("id")”可以省略。
2.restful.jsp
<head>
<title>RESTFUL测试</title>
<script type="text/javascript" src="${pageContext.request.contextPath }/js/jquery.min.js"></script>
<script type="text/javascript">
function search(){
// 获取输入的查询编号
var id = $("#number").val();
$.ajax({
url : "${pageContext.request.contextPath }/user/"+id,
type : "GET",
// 定义回调响应的数据格式为JSON字符串,该属性可以省略
dataType : "json",
success : function (data){
if(data.username != null){
alert("您查询的用户是:" + data.username);
}else{
alert("没有找到id为:"+id+"的用户");
}
}
});
}
</script>
</head>
<body>
<formr>
编号:<input type="text" name="number" id="number">
<input type="button" value="搜索" onclick="search()"/>
</formr>
</body>
【思考题】
1.请简述JSON数据交互两个注解的作用。
@responsebody放在方法上,用以返回
@requestparam放在参数上
2.请简述静态资源访问的几种配置方式。
<mvc: location=" " class=" ">
bean标签形式
servlet自带的