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,一经查实,立即删除!

相关文章

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

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

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

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

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

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

canvas绘制经典折线图(一)

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

win7笔记本为手机共享wifi

1、cmd netsh wlan set hostednetwork modeallow ssidyourname keyyourpassword 开启win7的虚拟wifi&#xff0c;让电脑变成无线路由器 这时&#xff0c;网络连接中会多出一个网卡为“Microsoft Virtual WiFi Miniport Adapter”的无线连接2。如果没有&#xff0c;需要更新无线…

createjs中shape的属性regX和regY

官方文档说regX和regY是图形与注册点的距离。 那么注册点是什么呢&#xff1f; 我理解注册点就是图形的x/y对应的点图形动效的原点就是注册点 如果修改图形的regX和regY值图形在画布上的位置是会被改变的&#xff0c;但是注册点其实并没有被改变。因为图形的x/y值并没有被改…

H3C 单区域OSPF配置示例二

转载于:https://www.cnblogs.com/fanweisheng/p/11163688.html

身份反模式:联邦筒仓和意大利面条身份

分析公司Quocirca的最新研究证实&#xff0c;现在许多企业的外部用户比内部用户更多&#xff1a;在欧洲&#xff0c;有58&#xff05;的企业直接与其他企业和/或消费者的用户进行交易&#xff1b; 仅在英国&#xff0c;这一数字就达到了65&#xff05;。 如果您回顾历史&#x…

多个气泡向上冒出!

这里展示白色半透明气泡如下图&#xff1a;实际是动态 思路&#xff1a;HTML里只需要一个CANVAS元素&#xff0c;Javascript里操作canvas 1、给canvas里绘制背景图片 2、在绘制半径为0-10px的圆形&#xff0c;x坐标屏幕水平随机&#xff0c;y所标竖直大于屏幕高度。 圆形背景…

分享25个优秀的网站底部设计案例

相对于网站头部来说&#xff0c;关注网站底部设计的人很少。我们平常也能碰到有些网站的底部设计得很漂亮&#xff0c;给网站的呈现来一个完美的结尾。这篇文章收集了25个优秀的网站底部设计案例&#xff0c;一起欣赏。 me & oli La Bubbly Poogan’s Porch GiftRocket Lin…

wbepack中output.filename和output.chunkFilename

对于webpack配置中filename和chunkFilename在使用中有些不懂的地方&#xff0c;研究之后记录如下。 filename: string | function 此选项决定了每个输出 bundle 的名称。这些 bundle 将写入到 output.path 选项指定的目录下。 对于单个入口起点&#xff0c;filename 会是一个…

OP AMP - 反馈理论在运放中的应用

实际应用的系统绝大多数是闭环的&#xff0c;运放更是如此。 关于反馈理论&#xff0c;需要了解极点&#xff0c;零点&#xff0c;波特图&#xff0c;以及如何用相位裕度&#xff0c;幅值裕度 确定系统的稳定性&#xff0c;这里不作赘述 &#xff0c;不懂的部分可以到网上搜点…

阅读react-redux源码 - 零

阅读react-redux源码 - 零阅读react-redux源码 - 一阅读react-redux源码(二) - createConnect、match函数的实现 react的技术栈一定会遇到redux&#xff0c;而在react中使用redux需要使用react-redux&#xff0c;那么react-redux是怎么包装redux已适用react的呢&#xff1f; …

CORS(跨域资源共享)

CORS&#xff08;跨域资源共享&#xff09;使用额外的HTTP头部来告诉浏览器&#xff0c;允许运行在origin(domain)上的Web应用访问来自不同源服务器上的指定资源。 浏览器访问一个web应用&#xff0c;这个web应用会发很多的跨域请求&#xff0c;例如加载不同源的JS/CSS脚本&am…

[转]jQuery设计思想

转自&#xff1a;http://kb.cnblogs.com/page/109875/ jQuery是目前使用最广泛的javascript函数库。 据统计&#xff0c;全世界排名前100万的网站&#xff0c;有46%使用jQuery&#xff0c;远远超过其他库。微软公司甚至把jQuery作为他们的官方库。 对于网页开发者来说&#xff…

敏捷开发绩效管理之四:为团队设立外部绩效目标(目标管理,外向型绩效)...

这是敏捷开发绩效管理的第四篇。&#xff08;之一&#xff0c;之二&#xff0c;之三&#xff0c;之四&#xff0c;之五&#xff0c;之六&#xff0c;之七&#xff09;最近在看德鲁克的书&#xff0c;发现其中很明确地写着“企业的绩效只存在于外部&#xff0c;而企业内部只有成…

【K8S in Action】服务:让客户端发现pod 并与之通信(2)

一 通过Ingress暴露服务 Ingress (名词&#xff09; 一一进入或进入的行为&#xff1b;进入的权利&#xff1b;进入的手段或地点&#xff1b;入口。一个重要的原因是每个 LoadBalancer 服务都需要自己的负载均衡器&#xff0c; 以及 独有的公有 IP 地址&#xff0c; 而 Ingres…

一个罐子统治一切:Apache TomEE + Shrinkwrap == JavaEE引导

警告&#xff1a;我不是Spring Boot的专家。 我发现很多事情对此非常有趣&#xff0c;并且当然可以真正改善您的日常工作。 而且&#xff0c;我对Spring Boot没有任何反对&#xff0c;也没有开发或使用它的人。 但是我认为社区高估了该产品。 一年前&#xff0c;我开始收到很多…

iview-admin框架运行步骤

第一步&#xff1a; 前往github下载整个iview-admin框架的全部源码 github地址&#xff1a; https://github.com/iview/iview-admin 第二步&#xff1a; 点击Clone or download绿色按钮。下载整个压缩包 第三步&#xff1a; 解压至D盘&#xff0c;在根目录中按 1、前往github下…

.net 垃圾回收学习[How To: Use CLR Profiler][翻译学习]【2】

http://msdn.microsoft.com/zh-cn/library/ms979205 注意&#xff1a;内容可能已经过期了。 注意&#xff1a;CLR Profiler最新版本&#xff1a;http://www.microsoft.com/download/en/details.aspx?id16273 Identifying Common Garbage Collection Issues 可以使用CLR Profil…