笔记67 Spring Boot快速入门(七)
SpringBoot+RESTful+JSON
一、RESTful架构
REST全称是Representational State Transfer,中文意思是表述(编者注:通常译为表征)性状态转移。 它首次出现在2000年Roy Fielding的博士论文中,Roy Fielding是HTTP规范的主要编写者之一。 他在论文中提到:"我这篇文章的写作目的,就是想在符合架构原理的前提下,理解和评估以网络为基础的应用软件的架构设计,得到一个功能强、性能好、适宜通信的架构。REST指的是一组架构约束条件和原则。" 如果一个架构符合REST的约束条件和原则,我们就称它为RESTful架构。
REST本身并没有创造新的技术、组件或服务,而隐藏在RESTful背后的理念就是使用Web的现有特征和能力, 更好地使用现有Web标准中的一些准则和约束。虽然REST本身受Web技术的影响很深, 但是理论上REST架构风格并不是绑定在HTTP上,只不过目前HTTP是唯一与REST相关的实例。 所以我们这里描述的REST也是通过HTTP实现的REST。
1.统一资源接口
RESTful架构应该遵循统一接口原则,统一接口包含了一组受限的预定义的操作,不论什么样的资源,都是通过使用相同的接口进行资源的访问。接口应该使用标准的HTTP方法如GET,PUT和POST,并遵循这些方法的语义。
如果按照HTTP方法的语义来暴露资源,那么接口将会拥有安全性和幂等性的特性,例如GET和HEAD请求都是安全的, 无论请求多少次,都不会改变服务器状态。而GET、HEAD、PUT和DELETE请求都是幂等的,无论对资源操作多少次, 结果总是一样的,后面的请求并不会产生比第一次更多的影响(幂等)。
下面列出了GET,DELETE,PUT和POST的典型用法:
| GET | DELETE | PUT | POST | 
| 
 
 
 | 
 
 
 | 
 
 
 | 
 
 
 | 
小结:
<1>POST和PUT用于创建资源时有什么区别?
POST和PUT在创建资源的区别在于,所创建的资源的名称(URI)是否由客户端决定。 例如为我的博文增加一个java的分类,生成的路径就是分类名/categories/java,那么就可以采用PUT方法。
<2>PATCH这种不是HTTP标准方法。
<3>统一资源接口对URI有什么指导意义?
统一资源接口要求使用标准的HTTP方法对资源进行操作,所以URI只应该来表示资源的名称,而不应该包括资源的操作。
通俗来说,URI不应该使用动作来描述。例如,下面是一些不符合统一接口要求的URI:
- GET /getUser/1
- POST /createUser
- PUT /updateUser/1
- DELETE /deleteUser/1
 
 
 
二、RESTful风格
简而言之,什么是REST? 就是看Url就知道要什么,看http method就知道干什么。
  在Web开发的过程中,method常用的值是get和post。可事实上,method值还可以是put和delete等等其他值。
既然method值如此丰富,那么就可以考虑使用同一个url,但是约定不同的method来实施不同的业务,这就是Restful的基本考虑。
CRUD是最常见的操作,在使用Restful 风格之前,通常的增加做法是这样的:/addCategory?name=xxx
可是使用了Restuful风格之后,增加就变成了:/category
CRUD如下表所示,URL就都使用一样的 "/category",区别只是在于method不同,服务器根据method的不同来判断浏览器期望做的业务行为。

1.CategoryController.java
CRUD的RequestMapping都修改为了/category,以前用的注解叫做@RequestMapper,现在分别叫做 GetMapper, PutMapper, PostMapper 和 DeleteMapper 用于表示接受对应的Method。
package com.example.springbootrestfuldemo.controller; import com.example.springbootrestfuldemo.dao.CategoryDAO;
import com.example.springbootrestfuldemo.pojo.Category;
import com.sun.org.apache.xpath.internal.operations.Mod;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.*; import java.util.List; @Controller
public class CategoryController {
@Autowired
CategoryDAO categoryDAO; //restful风格 //1.返回5条记录
@GetMapping("/category")
public String listCategory(Model model,@RequestParam(value = "start", defaultValue = "0") int start, @RequestParam(value = "size", defaultValue = "5") int size) throws Exception {
start = start<0?0:start;
Sort sort = new Sort(Sort.Direction.DESC, "id");
Pageable pageable = new PageRequest(start, size, sort);
Page<Category> page =categoryDAO.findAll(pageable);
model.addAttribute("page",page);
return "listCategories";
}
//2.保存一条记录
@PutMapping("/category")
public String addCategories(Category category) throws Exception {
System.out.println("保存一条记录");
categoryDAO.save(category);
return "redirect:/category";
}
//3.删除一条记录
@DeleteMapping("/category/{id}")
public String deleteCategory(Category category){
System.out.println("删除一条记录!");
categoryDAO.delete(category);
return "redirect:/category";
} //4.更新一条记录
@PostMapping("/category/{id}")
public String updateCategory(Category category){
System.out.println("更新一条记录!");
categoryDAO.save(category);
return "redirect:/category";
}
//6.跳转到编辑页
@GetMapping("/category/{id}")
public String addCategory(@PathVariable("id") int id, Model model) throws Exception{
System.out.println("编辑视图");
Category category=categoryDAO.getOne(id);
model.addAttribute("c",category);
return "editCategory";
} }
2.listCategories.html
<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
<title>Title</title>
</head>
<body>
<div class="showing">
<h2>springboot+jpa+thymeleaf+Restful</h2>
<div style="width:500px;margin:20px auto;text-align: center">
<table align="center" border="1" cellspacing="0">
<thead>
<tr>
<th>id</th>
<th>name</th>
<td>编辑</td>
<td>删除</td>
</tr>
</thead>
<tbody>
<tr th:each="c: ${page}">
<td align="center" th:text="${c.id}"></td>
<td align="center" th:text="${c.name}"></td>
<td align="center" ><a th:href="@{/category/{id}(id=${c.id})}">编辑</a></td>
<td>
<form th:action="@{/category/{id}(id=${c.id})}" action="/category" method="post">
<input type="hidden" name="_method" value="DELETE"/>
<button type="submit">删除</button>
</form>
</td>
</tr>
</tbody>
</table>
<br />
<div>
<a th:href="@{/category(start=0)}">[首 页]</a>
<a th:href="@{/category(start=${page.number -1})}">[上一页]</a>
<a th:if="${page.number!=page.totalPages -1}" th:href="@{/category(start=${page.number +1})}">[下一页]</a>
<a th:href="@{/category(start=${page.totalPages -1})}">[末 页]</a>
</div>
<form action="/category" method="post">
<!--修改提交方式为PUT-->
<input type="hidden" name="_method" value="PUT"/>
name:<input name="name"/><br/>
<button type="submit">提交</button>
</form> </div>
</div>
</body>
</html>
<1>增加
A:action修改为"/category"
B:增加如下<input type="hidden" name="_method" value="PUT">,虽然这个form的method是post, 但是spingboot看到这个_method的值是put后,会把其修改为put.
<form action="/category" method="post">
<!--修改提交方式为PUT-->
<input type="hidden" name="_method" value="PUT"/>
name:<input name="name"/><br/>
<button type="submit">提交</button>
</form>
<2>删除
原来的删除是通过超链接进行跳转,通过jQuery或者js可以将超链接的提交方式改变成成表单提交。然而我不会,所以用笨办法,<td>里面直接放一个form。
<td>
<form th:action="@{/category/{id}(id=${c.id})}" action="/category" method="post">
<input type="hidden" name="_method" value="DELETE"/>
<button type="submit">删除</button>
</form>
</td>
<3>修改
注意thymeleaf构建url的方式
 <td align="center" ><a th:href="@{/category/{id}(id=${c.id})}">编辑</a></td>
3.editCategory.html
修改action中的地址
<!DOCTYPE html>
<html lang="en" xmlns:th="http://www.thymeleaf.org">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
<title>Title</title>
</head>
<body>
<div class="showing">
<h2>springboot+jpa</h2> <div style="margin:0px auto; width:500px"> <form th:action="@{/category/{id}(id=${c.id})}" method="post"> name: <input name="name" th:value="${c.name}" /> <br/> <input name="id" type="hidden" th:value="${c.id}" />
<button type="submit">提交</button> </form>
</div>
</div>
</body>
</html>
4.完整代码:https://github.com/lyj8330328/springboot-restful-demo
三、分别是以json方式:提交,获取单个和获取多个
提交:http://localhost:8080/submit.html
获取单个:http://localhost:8080/getOne.html
获取多个:http://localhost:8080/getMany.html
基于Restful 风格的springboot进行修改。
1.修改Category.java
①增加个toString() 方便,便于显示
②增加个注解:@JsonIgnoreProperties({ "handler","hibernateLazyInitializer" }) ,否则会出错
package com.example.springbootjpademo.pojo; import com.fasterxml.jackson.annotation.JsonIgnoreProperties; import javax.persistence.*; @Entity
@Table(name = "category")
@JsonIgnoreProperties({ "handler","hibernateLazyInitializer" })
public class Category {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "id")
private int id; @Column(name = "name")
private String name;
public int getId() {
return id;
}
public void setId(int id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
} public String toString() {
return "Category [id=" + id + ", name=" + name + "]";
}
}
2.CategoryController.java
控制器里提供3个方法,分别用来处理json 提交,json获取单个对象,json获取多个对象。
package com.example.springbootjpademo.controller; import com.example.springbootjpademo.dao.CategoryDAO;
import com.example.springbootjpademo.pojo.Category;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.domain.Page;
import org.springframework.data.domain.PageRequest;
import org.springframework.data.domain.Pageable;
import org.springframework.data.domain.Sort;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.*; import java.util.List;
//@Controller
@RestController
public class CategoryController {
@Autowired
CategoryDAO categoryDAO; @GetMapping("/category")
public List<Category> listCategory(@RequestParam(value = "start", defaultValue = "0") int start,@RequestParam(value = "size", defaultValue = "5") int size) throws Exception {
start = start<0?0:start;
Sort sort = new Sort(Sort.Direction.DESC, "id");
Pageable pageable = new PageRequest(start, size, sort);
Page<Category> page =categoryDAO.findAll(pageable);
return page.getContent();
} @GetMapping("/category/{id}")
public Category getCategory(@PathVariable("id") int id) throws Exception {
Category c= categoryDAO.getOne(id);
System.out.println(c);
return c;
}
@PutMapping("/category")
public void addCategories(@RequestBody Category category) throws Exception {
Category category1=new Category();
category1.setName(category.getName());
categoryDAO.save(category1);
System.out.println("springboot接受到浏览器以JSON格式提交的数据:"+category);
}
}
3.submit.html
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
<title>用AJAX以JSON方式提交数据</title>
<script type="text/javascript" src="js/jquery.min.js"></script>
</head>
<body>
<form >
id:<input type="text" id="id" value="123" /><br/>
名称:<input type="text" id="name" value="category xxx"/><br/>
<input type="button" value="提交" id="sender" />
</form>
<div id="messageDiv"></div> <script>
$('#sender').click(function(){
var id=document.getElementById('id').value;
var name=document.getElementById('name').value;
var category={"name":name,"id":id};
var jsonData = JSON.stringify(category);
var page="category"; $.ajax({
type:"put",
url: page,
data:jsonData,
dataType:"json",
contentType : "application/json;charset=UTF-8",
success: function(result){
}
});
alert("提交成功,请在springboot控制台查看服务端接收到的数据"); });
</script>
</body> </html>
4.getOne.html
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<title>用AJAX以JSON方式获取数据</title>
<script type="text/javascript" src="js/jquery.min.js"></script>
</head>
<body>
<input type="button" value="通过AJAX获取一个Hero对象---" id="sender"> <div id="messageDiv"></div> <script>
$('#sender').click(function(){
var url="category/12";
$.get(
url,
function(data) {
console.log(data);
var json=data;
var name =json.name;
var id = json.id;
$("#messageDiv").html("分类id:"+ id + "<br>分类名称:" +name ); });
});
</script>
</body>
</html>
5.getMany.html
<!DOCTYPE html>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<title>用AJAX以JSON方式获取数据</title>
<script type="text/javascript" src="js/jquery.min.js"></script>
</head>
<body>
<input type="button" value="通过AJAX获取多个分类对象" id="sender"> <div id="messageDiv"></div> <script>
$('#sender').click(function(){
var url="category?start=0&size=100";
$.get(
url,
function(data) {
var categorys = data;
for(i in categorys){
var old = $("#messageDiv").html();
var category = categorys[i];
$("#messageDiv").html(old + "<br>"+category.id+" ----- "+category.name);
}
});
});
</script>
</body> </body>
</html>
6.直接调试,即采用AJAX的方式。也可采用Postman,结果如下:
<1>请求http://localhost:8080/category,应返回5条记录,默认分页大小为5。注意请求的方式必须是GET,否则会报错。

<2>请求http://localhost:8080/category/12,返回id=12的对象。

<3>获取全部数据http://localhost:8080/category?start=0&size=100

7.代码:https://github.com/lyj8330328/springboot-jpa-demo
四、总结
1.访问方式
通过http://localhost:8080/submit.html这种形式访问视图,必须将视图文件放入src/main目录下的webapp文件夹中,这样才能访问到视图,放在resources下的templates中就会出错。
2.注解
<1>@JsonIgnoreProperties({ "handler","hibernateLazyInitializer" })
因为懒加载这个对象属性只是一个代理对象,如果json直接当作一个存在的属性去序列化就会出现错误。
<2>@RestController
@RestController is a stereotype annotation that combines @ResponseBody and @Controller.
意思是:
@RestController注解相当于@ResponseBody + @Controller合在一起的作用。
1)如果只是使用@RestController注解Controller,则Controller中的方法无法返回视图,返回的内容就是Return 里的内容。
2)如果需要返回到指定页面,则需要用 @Controller配合视图解析器InternalResourceViewResolver才行。
3)如果需要返回JSON,XML或自定义mediaType内容到页面,则需要在对应的方法上加上@ResponseBody注解。
笔记67 Spring Boot快速入门(七)的更多相关文章
- 笔记61 Spring Boot快速入门(一)
		IDEA+Spring Boot快速搭建 一.IDEA创建项目 略 项目创建成功后在resources包下,属性文件application.properties中,把数据库连接属性加上,同时可以设置服 ... 
- 笔记65 Spring Boot快速入门(五)
		SpringBoot+JPA 一.什么是JPA? JPA是Java Persistence API的简称,中文名Java持久层API,是JDK 5.0注解或XML描述对象-关系表的映射关系,并将运行期 ... 
- 笔记63 Spring Boot快速入门(三)
		SpringBoot中使用JSP Springboot的默认视图支持是Thymeleaf,但是Thymeleaf还没开始学,熟悉的还是jsp,所以要让Springboot支持 jsp. 一.在pom. ... 
- 笔记70 Spring Boot快速入门(八)(重要)
		上传文件 一.方式一 1.上传页面 upLoadPage.html <!DOCTYPE html> <html lang="en"> <head> ... 
- 笔记64 Spring Boot快速入门(四)
		SpringBoot中错误处理.端口设置和上下文路径以及配置切换 一.错误处理 假设在访问首页的时候会出现一些错误,然后将这些错误当作异常抛出,反馈给用户. 1.修改IndexController.j ... 
- 笔记62 Spring Boot快速入门(二)
		SpringBoot部署 一.jar方式 1.首先安装maven. <1>下载最新的maven版本:https://maven.apache.org/download.cgi <2& ... 
- 笔记66 Spring Boot快速入门(六)
		SpringBoot中使用Mybatis 一.注解方式 1.创建映射文件CategoryMapper.java 使用注解@Mapper 表示这是一个Mybatis Mapper接口.使用@Select ... 
- Spring Boot 快速入门
		Spring Boot 快速入门 http://blog.csdn.net/xiaoyu411502/article/details/47864969 今天给大家介绍一下Spring Boot MVC ... 
- Spring Boot快速入门(二):http请求
		原文地址:https://lierabbit.cn/articles/4 一.准备 postman:一个接口测试工具 创建一个新工程 选择web 不会的请看Spring Boot快速入门(一):Hel ... 
随机推荐
- visulabox切换到菜单栏模式右ctrl + C
			2)首次看到1024x768的桌面时,查看可用的分辨率时,可能只能看到1024x768和800x600两种,其实,如果在virtualbox中按 右ctrl + C(Switch to Scaled ... 
- 编译lineageos1
			lineageos 前奏 -- 搭建编译环境 我目前使用的手机是红米note4x 目前lineageos15.1已经官方支持,下文是按照官网文档编译安装包操作总结 构建环境搭建主要参考官方文档 参考文 ... 
- 回收子进程——wait/waitpid 与 信号机制
			孤儿/僵尸进程——回收子进程 参考博客:https://blog.csdn.net/qq_35396127/article/details/78725915 :https://www.cnblogs. ... 
- JAVA读取Excel2003、2007、2010教程
			import java.io.File;import java.io.FileInputStream;import org.apache.poi.ss.usermodel.Row;import org ... 
- githup上传项目到仓库
			1.有了自己的账号 2.创建一个新的项目,填写项目名称,描述 填写完成点击create repository 3.复制生成的https链接接下来用到 4.进入到你的项目所在目录右键git bash打开 ... 
- python 常用技巧 — 字典 (dictionary)
			目录: 1. python 相加字典所有的键值 (python sum all values in dictionary) 2. python 两个列表分别组成字典的键和值 (python two l ... 
- Mysql中存储过程和函数的写法
			MySQL中,创建存储过程的基本形式如下: CREATE PROCEDURE sp_name ([proc_parameter[,...]]) [characteristic ...] routine ... 
- 集训队8月2日(BFS)
			看书情况:109~124页 刷题数:6 今天把上两次比赛的该补的题都补了,补题有博客,还写了两道书上例题的博客. 书上例题 BFS思维https://www.cnblogs.com/246247839 ... 
- Wireshark协议分析1
			一.界面简介 1.抓包工具栏 2.文件工具栏 3.包定位工具栏 4.颜色以及滚动界面工具栏 5.数据包列表字体定义工具栏 6.首选项工具栏 二.过滤规则 1.过滤 IP ip.src eq 192 ... 
- HTML中多媒体标签技术说明
			在纯文本的HTML页面中加入图片,给原来单调乏味的页面添加生气.HTML语言中利用<IMG>标记插入图片. 1.图片标记<IMG>及其属性 在网站上,网页设计者都使用了大量精心 ... 
