Spring Boot和Spring Data REST –通过REST公开存储库

logo-spring-io

使用Spring Boot和Spring Data REST,通过REST公开Spring Data存储库非常容易。 使用最少的代码,您可以创建遵循HATEOAS原理的JPA实体的REST表示。 我决定重用Spring PetClinic的JPA实体(业务层)作为本文的基础。

应用基础

PetClinic的模型相对简单,但是它包含一些单向和双向关联以及基本继承:

sbdr-petclinic图

此外,Spring的PetClinic为HSQLDB提供了SQL脚本,这使得在我的新应用程序中生成模式并用示例数据填充它非常容易。

项目依赖

作为配置的基础,我使用了Spring Initializr并生成了一个基本的Gradle项目。 为了在Spring Boot应用程序中利用Spring Data REST,我添加了以下Boot Starters:

compile("org.springframework.boot:spring-boot-starter-web")
compile("org.springframework.boot:spring-boot-starter-data-jpa")
compile("org.springframework.boot:spring-boot-starter-data-rest")

另外,我在项目中添加了HSQLDB依赖项:

compile("org.hsqldb:hsqldb:2.3.2")

原始项目使用org.joda.time.DateTime作为日期字段,并使用org.jadira.usertype.dateandtime.joda.PersistentDateTime将其与Hibernate持久化。 为了能够在新项目中使用它,我需要添加以下依赖项:

compile("joda-time:joda-time:2.4")
compile("org.jadira.usertype:usertype.jodatime:2.0.1")

使用API​​时,我注意到尽管原始项目中的date字段使用Spring的@DateTimeFormat进行了注释, @DateTimeFormat它们并未正确序列化。 我发现我需要使用@JsonFormatter ,因此另一个依赖项被添加到build.gradle

compile("com.fasterxml.jackson.datatype:jackson-datatype-joda:2.4.2");

进入类路径后,Spring Boot将通过org.springframework.boot.autoconfigure.jackson.JacksonAutoConfiguration自动配置com.fasterxml.jackson.datatype.joda.JodaModule

请注意,如果要正确序列化Java 8日期和时间类型,则需要向项目添加Jackson数据类型JSR310依赖项。

初始化数据库

为了初始化数据源,我在src/main/resources添加了schema-hsqldb.sqldata-hsqldb.sql文件。 最后,我将几个属性添加到application.properties

spring.datasource.platform = hsqldbspring.jpa.generate-ddl = falsespring.jpa.hibernate.ddl-auto = none

现在,在应用程序启动时,将自动拾取文件,并初始化数据源,并且由于有数据, 发现 API会容易得多!

储存库

Spring Data REST的一般思想是在Spring Data存储库的基础上构建并自动将其导出为REST资源 。 我创建了几个存储库,每个实体一个( OwnerRepositoryPetRepository等)。 所有存储库都是从PagingAndSortingRepository扩展的Java接口。

在此阶段不需要其他代码:不需要@Controller ,不需要配置(除非需要自定义)。 Spring Boot将自动为我们配置所有内容。

运行应用程序

有了整个配置,就可以执行项目(您将在文章底部找到指向整个项目的链接)。 如果幸运的话,该应用程序将启动,您可以导航到http://localhost:8080 ,它指向指向所有可用资源( 根资源 )的链接的集合。 响应的内容类型为。

哈尔

资源以Hypermedia样式实现,默认情况下,Spring Data REST使用HAL和内容类型为application/hal+json来呈现响应。 HAL是一种简单的格式,提供了一种链接资源的简便方法。 例:

$ curl localhost:8080/owners/1
{"firstName" : "George","lastName" : "Franklin","_links" : {"self" : {"href" : "http://localhost:8080/owners/1"},"pets" : {"href" : "http://localhost:8080/owners/1/pets"}}
}

就Spring Data REST而言,有几种类型的资源:集合,项目,搜索,查询方法和关联,并且都在响应中利用了application/hal+json内容类型。

馆藏和物品资源

集合资源支持GETPOST方法。 项目资源通常支持GETPUTPATCHDELETE方法。 注意, PATCH应用随请求主体发送的值,而PUT替换资源。

搜索和查找方法资源

搜索资源返回存储库公开的所有查询方法的链接,而查询方法资源执行通过存储库界面上的单个查询方法公开的查询。 两者都是只读的,因此仅支持GET方法。

为了可视化,我向OwnerRepository添加了find方法:

List<Owner> findBylastName(@Param("lastName") String lastName);

然后在http://localhost:8080/owners/search下公开:

$ curl http://localhost:8080/owners/search                                     
{                                                                              "_links" : {                                                                 "findBylastName" : {                                                       "href" : "http://localhost:8080/owners/search/findBylastName{?lastName}","templated" : true                                                       }                                                                          }                                                                            
}

协会资源

Spring Data REST自动公开子资源。 关联资源支持GETPOSTPUT方法。

并允许对其进行管理。 使用关联时,您需要了解文本/ uri-list内容类型。 具有这种内容类型的请求包含一个或多个URI( 每个URI应该出现在一行中,并且只有一行 )要添加到关联中的资源。

在第一个示例中,我们将研究Vet类中的单向关系:

@ManyToMany(fetch = FetchType.EAGER)
@JoinTable(name = "vet_specialties", joinColumns = @JoinColumn(name = "vet_id"),inverseJoinColumns = @JoinColumn(name = "specialty_id"))
private Set<Specialty> specialties;

为了将现有的专业添加到兽医的专业集合中,必须执行PUT请求:

curl -i -X PUT -H "Content-Type:text/uri-list" -d $'http://localhost:8080/specialties/1\nhttp://localhost:8080/specialties/2' http://localhost:8080/vets/1/specialties

可以使用DELETE方法删除关联,如下所示:

curl -i -X DELETE http://localhost:8080/vets/1/specialties/2

让我们看另一个例子:

// Owner
@OneToMany(mappedBy = "owner", cascade = CascadeType.ALL, orphanRemoval = true)
private Set<Pet> pets;// Pet
@ManyToOne(cascade = CascadeType.ALL, optional = false)
@JoinColumn(name = "owner_id")
private Owner owner;

设置宠物主人可以通过以下请求完成:

curl -i -X PUT -H "Content-Type:text/uri-list" -d "http://localhost:8080/owners/1" http://localhost:8080/pets/2/owner

但是如何移除所有者呢? 由于必须始终为宠物设置所有者,因此在尝试使用以下命令取消设置时,我们会收到HTTP/1.1 409 Conflict

curl -i -X DELETE http://localhost:8080/pets/2/owner

整合测试

使用Spring Boot,可以在测试中启动Web应用程序,并使用Spring Boot的@IntegrationTest对其进行验证。 我们将使用RestTemplate及其Spring Boot的实现来验证实际的REST调用,而不是使用MockMvc服务器端Web应用程序上下文( MockMvc )。

众所周知,资源的类型为application/hal+json 。 因此实际上,将它们直接反序列化为实体对象(例如Owner )是不可能的。 相反,必须将其反序列化为org.springframework.hateoas.Resource ,以包装实体并为其添加链接。 而且,由于Resource是一个通用型ParameterizedTypeReference必须使用RestTemplate

以下示例将其可视化:

private RestTemplate restTemplate = new TestRestTemplate();@Test
public void getsOwner() {String ownerUrl = "http://localhost:9000/owners/1";ParameterizedTypeReference<Resource<Owner>> responseType = new ParameterizedTypeReference<Resource<Owner>>() {};ResponseEntity<Resource<Owner>> responseEntity =restTemplate.exchange(ownerUrl, GET, null, responseType);Owner owner = responseEntity.getBody().getContent();assertEquals("George", owner.getFirstName());// more assertions}

下一篇文章对此方法进行了很好的描述: 使用Spring RestTemplate和Super类型令牌使用Spring-hateoas Rest服务

摘要

通过几个步骤以及Spring Boot和Spring Data REST的强大功能,我为现有PetClinic的数据库创建了API。 使用Spring Data REST可以做更多的事情(例如自定义),除了文档不多,与其他Spring项目相比,Spring Data REST似乎可以大大加快开发速度。 我认为,这是一个需要快速原型制作的好项目。

参考文献

  • 源代码
    • GitHub上的Spring Boot PetClinic API
  • 说明文件:
    • Spring Data REST
  • 文章:
    • 恢复您的JPA实体

翻译自: https://www.javacodegeeks.com/2014/10/spring-boot-and-spring-data-rest-exposing-repositories-over-rest.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/362061.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

记录一次cookie导致登录失败的惨案

现象描述&#xff1a; 前端登录成功后并没有从后端那里拿到登录信息&#xff0c;换句话说登录服务告诉我们登录成功了&#xff0c;但是后端却说我们没有登录成功。 背景&#xff1a; 因为前后端分离&#xff0c;所以前后端项目部署在两个子域名下。 因为要打通登录态&#…

sql中的遇到的有问题的

----题5&#xff1a;求出住在同一城市的顾客对--select city,count(cid) as 顾客的个数 from customers group by city--select --select distinct c1.cname,c2.cname,c1.city from customers c1, customers c2 where c1.cityc2.city and c1.cname<c2.cname select c1.cid,c…

nyoj 55 懒省事的小明 优先队列 multiset 还有暴力

懒省事的小明 时间限制&#xff1a; 3000 ms | 内存限制&#xff1a; 65535 KB难度&#xff1a; 3描述小明很想吃果子&#xff0c;正好果园果子熟了。在果园里&#xff0c;小明已经将所有的果子打了下来&#xff0c;而且按果子的不同种类分成了不同的堆。小明决定把所有的果子…

css3之3d导航

css3的新属性非常不错&#xff0c;目前IE除外其他浏览器都已支持 实现原理:比如元素a在hover时候可以改变元素b的状态。 效果如本农导航&#xff0c;欢迎采用和建议~ a:hover b{ 执行简单动画效果 } HTML <!DOCTYPE html><html lang"en"><head&…

jQuery Ajax – Servlets集成:构建完整的应用程序

网上有很多教程&#xff0c;它们解释了有关使用servlet和JSP页面进行Java Web开发的一些知识&#xff0c;但是&#xff0c;我从来没有找到对于初学者来说足够简洁&#xff0c;简单的教程。 这样的教程应该解释创建一个简单的Web应用程序的整个过程&#xff0c;包括前端&#xf…

ES6中块级作用域下的函数声明

背景 因为ES5的时候没有块级作用域&#xff0c;所以ES5规定不能再if这样的块中声明函数&#xff0c;但是为了兼容各大浏览器并没有严格遵守这条规定。 ES6的时候引入了块级作用域&#xff0c;规定在块级作用域中声明函数就相当于使用let来声明变量一样。但是又因为浏览器端的…

Oracle数据库卸载

Oracle数据库卸载 ORACLE数据库安装起来比较麻烦&#xff0c;卸载也不像微软的产品那样容易。对于ORACLE9的卸载&#xff0c;控制面板里是没有卸载程序的。可以从开始菜单—程序—Oracle Installation Products—Universal Installer 进入安装的界面&#xff0c;界面上有一个…

信息系统开发平台OpenExpressApp - 应用模型ApplicationModel

下图为OpenExpressApp的系统架构图&#xff0c;其中在应用模型是作为一种元数据贯穿于整个架构&#xff0c;应用模型运行在OpenExpressApp框架之上。应用模型是OEA的核心&#xff0c;理解好应用模型才能更好的使用OEA。 应用模型贯穿于整个架构层 模型关注what OEA希望从重复的…

SVN版本控制系统学习(中文配置)

先吐槽下往上搜索的一些SVN下载地址&#xff0c;里面乱七八糟啥都有&#xff0c;下载好后点击安装一不注意啥玩意都安装上了&#xff0c; 什么玩意都有&#xff0c;真心不明白这些推送者是怎么想的。搜集了一个WIN32的百度网盘下载地址&#xff1a; http://pan.baidu.com/s/1…

【译】nginx关于location部分

译&#xff1a; Syntax:location [ | ~ | ~* | ^~ ] uri { ... }location name { ... }Default:—Context:server, location 依据请求的URI进行配置。 在对以"%xx"形式的文本解码&#xff0c;对相对路径".“和”…"的格式化和两个或多个相邻斜杠压缩为单…

自定义汇编程序,Weaver和运行时的可插拔知识

作为贝叶斯工作的一部分&#xff0c;我对Kie进行了很多重构&#xff0c;使其具有清晰的扩展点。 我想确保可以完成贝叶斯系统的所有工作部件&#xff0c;而无需在现有内核中添加任何代码。 因此&#xff0c;现在每种知识类型都可以拥有自己的包&#xff0c;汇编器&#xff0c;…

动态代理

java动态代理 JAVA的动态代理 代理模式 代理模式是常用的java设计模式&#xff0c;他的特征是代理类与委托类有同样的接口&#xff0c;代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类&#xff0c;以及事后处理消息等。代理类与委托类之间通常会存在关联关系&…

Kafka学习(一)-------- Quickstart

参考官网&#xff1a;http://kafka.apache.org/quickstart 一、下载Kafka 官网下载地址 http://kafka.apache.org/downloads 截至2019年7月8日 最新版本为 2.3.0 2.12为编译的scala版本 2.3.0为kafka版本 Scala 2.12 - kafka_2.12-2.3.0.tgz (asc, sha512) 解压 > tar -xzf…

关于页面布局间距使用的经验之谈

在页面布局的时候遇到一个问题在此记录。 有如下布局需求。页面上大多数都是这样的&#xff0c;一块一块从上到下排列。 块与块之间的间距需要固定由谁来负责。例如第一个块和第二个块之间的间距&#xff0c;就需要第二个块的margin-top完成&#xff0c;文字和第二个块之间的间…

最简洁的js鼠标拖曳效果【原】

请原谅我是一个标题档&#xff0c;不过还是很简洁的&#xff0c;因为只是初步的实现的拖曳效果<!DOCTYPE html><html><head><meta http-equiv"Content-Type"content"text/html; charsetutf-8"/><meta http-equiv"Content-…

safari 音频播放问题

问题描述&#xff1a; 点击播放音频按钮发现并没有声音&#xff08;并不是自动播放&#xff0c;是有用户行为的&#xff09;。 import React, { useEffect, useState, useRef } from reactfunction comp() {let [paused, setPaused] useState(true)let audioDom useRef(null…

canvas绘制经典折线图(一)

最终效果图如下&#xff1a; 实现步骤如下&#xff1a;注-引用了jQuery HTML代码 <!doctype html><html lang"en"><head><meta charset"UTF-8"><meta name"Generator" content"EditPlus"><meta nam…

如何从Java EE无状态应用程序连接到MongoDB

在本文中&#xff0c;我将介绍如何从无状态Java EE应用程序连接到MongoDB&#xff0c;以利用与MongoDB Java驱动程序提供的数据库的内置连接池。 如果您开发的REST API对MongoDB执行操作&#xff0c;则可能是这种情况。 获取Java MongoDb驱动程序 要将Java连接到MongoDB&#…

开发流程补全

在开发过程中我意识到一个问题 具体问题就是我没有一个可靠的机制来防止自己犯错 现在的流程是 开发 调试 -> 测试同学 -> 上线 这里测试的时间会有点长&#xff0c;因为bug会有点多&#xff0c;然后需要修改bug&#xff0c;然后测试验证 改bug时间 理解测试bug描述…

Linux 锁机制

本文讨论了 Linux 内核中可用的大量同步或锁定机制。这些机制为 2.6 版内核的许多可用方法提供了应用程序接口&#xff08;API&#xff09; 。但是在深入学习 API 之前&#xff0c;首先需要明白将要解决的问题。 当存在并发特性时&#xff0c;必须使用同步方法。当在同一时间段…