带有Hibernate OGM的NoSQL –第一部分:持久化您的第一个实体

Hibernate OGM的第一个最终版本已经发布 ,团队从发布狂潮中恢复了一些。 因此,他们考虑建立一系列教程式博客,使您有机会轻松地从Hibernate OGM重新开始。 感谢Gunnar Morling( @gunnarmorling )创建了本教程。

介绍

不知道什么是Hibernate OGM? Hibernate OGM是Hibernate下的最新项目,它允许您通过著名的JPA将实体模型持久保存在不同的NoSQL存储中。

我们将在接下来的几周内介绍这些主题:

  • 坚持您的第一个实体(此批次)
  • 查询数据
  • 在WildFly上运行
  • 在Java SE上使用CDI运行
  • 将数据存储到同一应用程序中的两个不同存储中

如果您希望我们讨论其他主题,请告诉我们。 只需在下面添加评论或向我们发送您的建议 。

在本系列的第一部分中,我们将建立一个具有所需依赖项的Java项目,创建一些简单实体,然后将它们写入商店或从商店中读取。 我们将从Neo4j图形数据库开始 ,然后仅需进行少量配置更改就切换到MongoDB文档存储 。

项目设置

首先让我们创建一个具有所需依赖项的新Java项目。 在下面,我们将使用Maven作为构建工具,但是Gradle或其他工具当然可以很好地工作。

将其添加到pom.xml的dependencyManagement块中:

...
<dependencyManagement><dependencies>...<dependency><groupId>org.hibernate.ogm</groupId><artifactId>hibernate-ogm-bom</artifactId><type>pom</type><version>4.1.1.Final</version><scope>import</scope></dependency>...</dependencies>
</dependencyManagement>
...

这将确保您使用的是匹配版本的Hibernate OGM模块及其依赖项。 然后将以下内容添加到dependencies块:

...
<dependencies>...<dependency><groupId>org.hibernate.ogm</groupId><artifactId>hibernate-ogm-neo4j</artifactId></dependency><dependency><groupId>org.jboss.jbossts</groupId><artifactId>jbossjta</artifactId></dependency>...
</dependencies>
...

依赖项是:

  • Hibernate OGM模块,用于处理嵌入式Neo4j数据库; 这将引入所有其他必需的模块,例如Hibernate OGM内核和Neo4j驱动程序。 使用MongoDB时,可以将其与hibernate-ogm-mongodb交换。
  • JBoss对Java事务API(JTA)的实现,当不在WildFly之类的Java EE容器中运行时需要此实现

领域模型

我们的示例域模型由三个类组成: HikeHikeSectionPerson

027c5fc5

之间存在组合关系HikeHikeSection ,即上调包括几个部分,其生命周期是完全依赖于远足。 远足部分列表已订购; 坚持远足及其部分时,必须保持此顺序。

HikePerson (充当远足组织者)之间的关联是双向的多对一/一对多关系:一个人可以组织零个或多个远足,而一个远足恰好有一个人充当组织者。

映射实体

现在,通过创建实体类并用所需的元数据注释它们来映射域模型。 让我们从Person类开始:

@Entity
public class Person {@Id@GeneratedValue(generator = "uuid")@GenericGenerator(name = "uuid", strategy = "uuid2")private long id;private String firstName;private String lastName;@OneToMany(mappedBy = "organizer", cascade = CascadeType.PERSIST)private Set<Hike> organizedHikes = new HashSet<>();// constructors, getters and setters...
}

实体类型使用@Entity注释进行标记,而表示标识符的属性则使用@Id注释。

Hibernate OGM无需手动分配ID,而是提供了几种ID生成策略,例如(模拟)序列,UUID等,因此可以解决这一问题。 通常,使用UUID生成器是一个不错的选择,因为它可以确保跨不同NoSQL数据存储的可移植性,并使ID生成快速且可扩展。 但是根据您使用的商店,在MongoDB的情况下,您也可以使用特定的ID类型,例如对象ID(有关详细信息,请参阅参考指南 )。

最后, @OneToManyorganizedHikes属性标记为实体之间的关联。 由于它是双向实体,因此必须使用mappedBy属性来指定负责管理它的关联方。 指定级联类型PERSIST可确保坚持一个人也将自动导致其关联的远足也得以坚持。

接下来是Hike类:

@Entity
public class Hike {@Id@GeneratedValue(generator = "uuid")@GenericGenerator(name = "uuid", strategy = "uuid2")private String id;private String description;private Date date;private BigDecimal difficulty;@ManyToOneprivate Person organizer;@ElementCollection@OrderColumn(name = "sectionNo")private List<HikeSection> sections;// constructors, getters and setters...
}

在这里, @ManyToOne批注标记了HikeOrganizer之间的双向关联的另一面。 由于HikeSection应该依赖于Hike,因此节列表通过@ElementCollection映射。 为了确保在数据存储区中保持节的顺序, @OrderColumn使用@OrderColumn 。 这将在保留的记录中添加一个额外的“列”,其中包含每个部分的订单号。

最后, HikeSection类:

@Embeddable
public class HikeSection {private String start;private String end;// constructors, getters and setters...
}

PersonHike不同,它不是通过@Entity而是使用@Embeddable映射的。 这意味着它始终是另一个实体(在本例中为Hike )的一部分,因此也没有自己的身份。 因此,它没有声明任何@Id属性。

请注意,如果您一直在将Hibernate ORM与关系数据存储一起使用,则这些映射看起来完全相同。 的确,这就是Hibernate OGM的承诺之一:尽可能简化关系和NoSQL范例之间的迁移!

创建persistence.xml

有了实体类之后,JPA的persistence.xml描述符又丢失了一件事。 在src / main / resources / META-INF / persistence.xml下创建它:

<?xml version="1.0" encoding="utf-8"?><persistence xmlns="http://java.sun.com/xml/ns/persistence"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://java.sun.com/xml/ns/persistence http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd"version="2.0"><persistence-unit name="hikePu" transaction-type="RESOURCE_LOCAL"><provider>org.hibernate.ogm.jpa.HibernateOgmPersistence</provider><properties><property name="hibernate.ogm.datastore.provider" value="neo4j_embedded" /><property name="hibernate.ogm.datastore.database" value="HikeDB" /><property name="hibernate.ogm.neo4j.database_path" value="target/test_data_dir" /></properties></persistence-unit>
</persistence>

如果您以前使用过JPA,那么您对持久性单元的定义应该非常熟悉。 与在关系数据库之上使用经典Hibernate ORM的主要区别在于,我们需要为Hibernate OGM指定特定的提供程序类: org.hibernate.ogm.jpa.HibernateOgmPersistence

此外,还定义了一些特定于Hibernate OGM和所选后端的属性来设置:

  • 使用的后端(在这种情况下为嵌入式Neo4j图形数据库)
  • Neo4j数据库的名称
  • 存储Neo4j数据库文件的目录

根据您的使用情况和后端,可能需要其他属性,例如用于设置主机,用户名,密码等。您可以在名为<BACK END>Properties的类中找到所有可用的<BACK END>Properties ,例如Neo4jProperties , MongoDBProperties等。 。

保存和加载实体

有了所有这些位之后,就可以保留(并加载)某些实体。 为此创建一个简单的JUnit测试外壳:

public class HikeTest {private static EntityManagerFactory entityManagerFactory;@BeforeClasspublic static void setUpEntityManagerFactory() {entityManagerFactory = Persistence.createEntityManagerFactory( "hikePu" );}@AfterClasspublic static void closeEntityManagerFactory() {entityManagerFactory.close();}
}

这两种方法为persistence.xml中定义的持久性单元管理实体管理器工厂。 它保存在一个字段中,因此可以用于多种测试方法(请记住,实体管理器工厂创建起来相当昂贵,因此应将它们初始化一次并保留以备重复使用)。

然后创建一个持久保存并加载一些数据的测试方法:

@Test
public void canPersistAndLoadPersonAndHikes() {EntityManager entityManager = entityManagerFactory.createEntityManager();entityManager.getTransaction().begin();// create a PersonPerson bob = new Person( "Bob", "McRobb" );// and two hikesHike cornwall = new Hike("Visiting Land's End", new Date(), new BigDecimal( "5.5" ),new HikeSection( "Penzance", "Mousehole" ),new HikeSection( "Mousehole", "St. Levan" ),new HikeSection( "St. Levan", "Land's End" ));Hike isleOfWight = new Hike("Exploring Carisbrooke Castle", new Date(), new BigDecimal( "7.5" ),new HikeSection( "Freshwater", "Calbourne" ),new HikeSection( "Calbourne", "Carisbrooke Castle" ));// let Bob organize the two hikescornwall.setOrganizer( bob );bob.getOrganizedHikes().add( cornwall );isleOfWight.setOrganizer( bob );bob.getOrganizedHikes().add( isleOfWight );// persist organizer (will be cascaded to hikes)entityManager.persist( bob );entityManager.getTransaction().commit();// get a new EM to make sure data is actually retrieved from the store and not Hibernate's internal cacheentityManager.close();entityManager = entityManagerFactory.createEntityManager();// load it backentityManager.getTransaction().begin();Person loadedPerson = entityManager.find( Person.class, bob.getId() );assertThat( loadedPerson ).isNotNull();assertThat( loadedPerson.getFirstName() ).isEqualTo( "Bob" );assertThat( loadedPerson.getOrganizedHikes() ).onProperty( "description" ).containsOnly( "Visiting Land's End", "Exploring Carisbrooke Castle" );entityManager.getTransaction().commit();entityManager.close();
}

注意这两个动作在事务中如何发生。 Neo4j是一个完全事务性的数据存储库,可以通过JPA的事务处理API很好地控制它。 在实际的应用程序中,可能会使用一种较不冗长的方法进行事务控制。 根据所选的后端和应用程序运行的环境类型(例如Java EE容器,例如WildFly ),您可以利用通过CDI或EJB进行的声明式事务管理。 但是,让我们再保存一次。

保留了一些数据后,您可以使用Neo4j随附的漂亮的Web控制台对其进行检查。 下面显示了测试保留的实体:

aDXbhi6

Hibernate OGM旨在为您要定位的数据存储提供最自然的映射。 对于Neo4j作为图形数据存储的情况,这意味着任何实体都将映射到相应的节点。

实体属性被映射为节点属性(请参见描述黑名单节点之一的黑框)。 任何不受本机支持的属性类型都将根据需要进行转换。 例如, date属性就是这种情况,它以ISO格式的字符串形式保留。 此外,每个实体节点都具有标签 ENTITY(以将其与其他类型的节点区分开)和用于指定其实体类型的标签(在这种情况下为Hike)。

关联被映射为节点之间的关系,关联角色被映射为关系类型 。

请注意,Neo4j没有嵌入式对象的概念。 因此,将HikeSection对象映射为带有标签EMBEDDED的节点,并与拥有的Hike节点链接。 节的顺序通过关系上的属性保留。

切换到MongoDB

Hibernate OGM的承诺之一是允许使用相同的API(即JPA)与不同的NoSQL存储一起使用。 因此,让我们看看MongoDB是如何保存和使用的,与Neo4j不同,MongoDB是一个文档数据存储区,并以类似JSON的表示形式保留数据。 为此,请首先将Neo4j后端替换为以下内容:

...
<dependency><groupId>org.hibernate.ogm</groupId><artifactId>hibernate-ogm-mongodb</artifactId>
</dependency>
...

然后使用可通过以下方式访问的属性更新persistence.xml中的配置,以将MongoDB作为后端使用
提供与您的环境匹配的主机名和凭据的MongoDBProperties (如果尚未安装MongoDB,则可以在此处下载):

...
<properties><property name="hibernate.ogm.datastore.provider" value="mongodb" /><property name="hibernate.ogm.datastore.database" value="HikeDB" /><property name="hibernate.ogm.datastore.host" value="mongodb.mycompany.com" /><property name="hibernate.ogm.datastore.username" value="db_user" /><property name="hibernate.ogm.datastore.password" value="top_secret!" />
</properties>
...

这就是将实体保留在MongoDB中而不是Neo4j中所需要做的全部工作。 如果现在再次运行测试,您将在数据存储区中找到以下BSON文档:

# Collection "Person"
{"_id" : "50b62f9b-874f-4513-85aa-c2f59015a9d0","firstName" : "Bob","lastName" : "McRobb","organizedHikes" : ["a78d731f-eff0-41f5-88d6-951f0206ee67","32384eb4-717a-43dc-8c58-9aa4c4e505d1"]
}
# Collection Hike
{"_id" : "a78d731f-eff0-41f5-88d6-951f0206ee67","date" : ISODate("2015-01-16T11:59:48.928Z"),"description" : "Visiting Land's End","difficulty" : "5.5","organizer_id" : "50b62f9b-874f-4513-85aa-c2f59015a9d0","sections" : [{"sectionNo" : 0,"start" : "Penzance","end" : "Mousehole"},{"sectionNo" : 1,"start" : "Mousehole","end" : "St. Levan"},{"sectionNo" : 2,"start" : "St. Levan","end" : "Land's End"}]
}
{"_id" : "32384eb4-717a-43dc-8c58-9aa4c4e505d1","date" : ISODate("2015-01-16T11:59:48.928Z"),"description" : "Exploring Carisbrooke Castle","difficulty" : "7.5","organizer_id" : "50b62f9b-874f-4513-85aa-c2f59015a9d0","sections" : [{"sectionNo" : 1,"start" : "Calbourne","end" : "Carisbrooke Castle"},{"sectionNo" : 0,"start" : "Freshwater","end" : "Calbourne"}]
}

同样,映射是非常自然的,就像您在使用MongoDB这样的文档存储时所期望的那样。 通过在任一侧存储引用的ID,可以映射PersonHike之间的双向一对多/多对一关联。 当加载回数据时,Hibernate OGM将解析ID,并允许将关联从一个对象导航到另一个对象。

元素集合使用MongoDB的存储分层结构的功能进行映射。 在此,加息的各个部分映射到拥有加息的文档中的数组,并带有附加的字段sectionNo来维护收集顺序。 这样可以通过一次往返数据存储的方式非常有效地加载实体及其嵌入式元素。

包起来

带有Hibernate OGM 101NoSQL的第一部分中,您学习了如何设置具有所需依赖项的项目,映射某些实体和关联并将其持久化在Neo4j和MongoDB中。 所有这些都是通过众所周知的JPA API进行的。 因此,如果您过去在关系数据库之上使用过Hibernate ORM和JPA,那么深入NoSQL领域将变得异常简单。

同时,每个商店都针对某些用例,因此提供了特定的功能和配置选项。 自然,这些不能通过通用API(例如JPA)公开。 因此,Hibernate OGM允许您使用本机NoSQL查询,并允许通过其灵活的选项系统配置商店特定的设置。

您可以在GitHub上找到此博客文章的完整示例代码 。 只需叉子,然后随心所欲地玩就可以了。

当然,存储实体并通过其ID取回它们仅仅是开始。 在任何实际的应用程序中,您都希望对数据运行查询,并且您可能还想利用所选NoSQL存储的某些特定功能和设置。 我们将在本系列的下一部分中谈到这一点,请继续关注!

翻译自: https://www.javacodegeeks.com/2015/01/nosql-with-hibernate-ogm-part-one-persisting-your-first-entities.html

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

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

相关文章

为自己写程序之JavsScript代码段测试器

JavaScript的测试&#xff0c;通常是在Firefox的firebug插件中测试的。不过有时只是测试几行代码都要写一个html&#xff0c;再打开浏览器测试运行结果&#xff0c;感觉并不是很方便。 今天花了点时间做了一个简易的JS片段测试器。其实这主要是看了IronJs开源项目以后&#xff…

sizeof和strlen的区别(其中涉及NUL的讲解)

本文是自己结合平时所学的知识&#xff0c;对sizeof和strlen的区别进行了总结&#xff0c;如有不对的地方还请批评指证&#xff0c;共同进步&#xff01;&#xff01;&#xff01; 一、从C语言的定义上来讲 1、sizeof是关键字&#xff0c;而strlen是包含在string.h头文件中的一…

使用WildFly和Java EE 7映像与Docker提供者一起流浪

什么是无业游民&#xff1f; Vagrant是创建虚拟开发环境的简化且可移植的方式。 它可与多种虚拟化软件一起使用&#xff0c;例如VirtualBox&#xff0c;VMWare&#xff0c;AWS等。 它还可以与多种配置软件一起使用&#xff0c;例如Ansible&#xff0c;Chef&#xff0c;Puppet或…

休眠锁定模式–乐观锁定模式如何工作

显式乐观锁定 在上一篇文章中 &#xff0c;我介绍了Java持久性锁定的基本概念。 隐式锁定机制可防止丢失更新 &#xff0c;它适用于我们可以主动修改的实体。 虽然隐式乐观锁定是一种广泛使用的技术&#xff0c;但是很少有人了解显式乐观锁定模式的内部工作原理。 当锁定的实…

如何在JMeter中执行客户端Web性能测试?

在本文中&#xff0c;我们将看到如何使用Jmeter插件进行客户端性能测试。 我将使用jmeter webdriver插件。 在开始本主题之前&#xff0c;请从我以前的文章中获得有关客户端性能测试的一些基本信息。 因此&#xff0c;让我们开始吧&#xff1a; 安装 通过这篇文章之后的链接&…

[转载]Buffon投针实验:究竟为什么是pi?

Buffon投针实验&#xff1a;究竟为什么是pi&#xff1f; Brain Storm | 2009-11-06 20:16| 57 Comments | 本文内容遵从CC版权协议 转载请注明出自matrix67.com数学学习真正悲哀的就是&#xff0c;记住了某个神奇而伟大的定理&#xff0c;看懂了其最严密的推导过程&#xff0c;…

初始化懒惰关系以及何时使用它们的5种方法

实体之间关系的延迟加载是JPA中公认的最佳实践。 它的主要目标是仅从数据库中检索请求的实体&#xff0c;并仅在需要时加载相关实体。 如果我们只需要请求的实体&#xff0c;那是一个很好的方法。 但是&#xff0c;如果我们还需要一些相关实体&#xff0c;它会增加工作量&#…

fieldset ----- 不常用的HTML标签

fieldset 元素可将表单内的相关元素分组。 <fieldset> 标签将表单内容的一部分打包&#xff0c;生成一组相关表单的字段。 当一组表单元素放到 <fieldset> 标签内时&#xff0c;浏览器会以特殊方式来显示它们&#xff0c;它们可能有特殊的边界、3D 效果&#xff…

使用入站适配器公开HTTP Restful API。 第1部分(XML)

1.简介 这篇文章的目的是使用Spring Integration HTTP入站适配器实现HTTP Restful API。 本教程分为两个部分&#xff1a; XML配置示例&#xff08;同一篇文章&#xff09;。 Java DSL示例。 这将在本教程的下一部分中进行说明&#xff0c;展示如何使用Spring Integration Ja…

使用jOOQ和JavaFX将SQL数据转换为图表

最近&#xff0c;我们已经展示了Java 8和函数式编程将如何为使用jOOQ和Java 8 lambda和Streams进行SQL数据的函数数据转换为Java开发人员带来新的视角。 今天&#xff0c;我们将这一步骤更进一步&#xff0c;将数据转换为JavaFX XYChart.Series以根据数据生成美观的条形图。 设…

node.js学习笔记(1)

一&#xff0e; 安装以及环境配置 安装路径 http://nodejs.cn/download/ 多种环境选择 环境变量的配置 Step1 先检查环境变量中的系统变量里面的path,查看是否加入了node.js 例如我的node.js安装路径是C:\Program Files\nodejs 那么&#xff0c;这个path里面就应该加…

主要版本发布后Java开发人员应使用的15种工具

新部署的生存工具包&#xff1a;适用于Java开发人员的工具&#xff0c;这些工具经常将代码部署到生产中&#xff01; Takipi会检测生产中的所有错误&#xff0c;并像发生错误时一样显示变量值 立即部署并获得免费的T恤 适用于新部署的终极生存套件 与在僵尸末日场景下玩弄&…

Java EE 7批处理和魔兽世界–第2部分

今天&#xff0c;我将把第二部分带到我以前关于Java EE 7批处理和《魔兽世界–第1部分》的帖子中。 在本文中&#xff0c;我们将了解如何从第1部分中获得的数据中汇总和提取指标。 概括 批处理目的是下载魔兽世界拍卖行的数据&#xff0c;处理拍卖并提取指标。 这些指标将建立…

js导航条 二级滑动 模仿块级作用域

for(var i 1;i<7;i){    //因为首级标题有6个&#xff0c;对每个首级标题添加mouseover和mouseout事件。    //这里用到块级作用域(function(k){document.getElementById("p_"k).addEventListener(mouseover,function(event){document.getElementById(p_…

struts+swfupload实现批量图片上传(上):swfupload

custom_settings : {progressTarget : "fsUploadProgress",cancelButtonId : "btnCancel",uploadButtonId : "btnUpload",myFileListTarget : "idFileList" },custom_settings调用方法 this.customSettings.cancelButtonId 缩略图js …

40行中的持久性KeyValue Server和一个可悲的事实

再次出现。. 回顾 Peters关于Unsafe用法的书面概述 &#xff0c;我将简要介绍一下Java中的低级技术如何通过启用更高级别的抽象或允许Java性能级别来节省开发工作可能很多人都不知道。 我的主要观点是表明&#xff0c;将对象转换为字节&#xff0c;反之亦然是一个重要的基础&a…

TreeMap源码分析——深入分析(基于JDK1.6)

TreeMap有Values、EntrySet、KeySet、PrivateEntryIterator、EntryIterator、ValueIterator、KeyIterator、DescendingKeyIterator、NavigableSubMap、AscendingSubMap、DescendingSubMap、SubMap、Entry共十三个内部类。Entry是在TreeMap中用于表示树的节点的内部类&#xff0…

Python2.6 Cx_Oracle Linux下编译安装

分类&#xff1a; python Oracle 2012-06-07 00:04 239人阅读 评论(0) 收藏 举报(一) Python 2.6 安装 1.下载Python2.6.X 版本的源码包&#xff0c;这里采用平台编译安装。 Python-2.6.4.tar.bz2 2.解压缩 ,使用J参数解压bigz2类型的压缩文件 tar -jxvf Python-2.6.4.tar.bz2…

Apache TomEE(和Tomcat)的自签名证书

可能在大多数Java EE项目中&#xff0c;您将拥有具有SSL支持&#xff08; https &#xff09;的部分或整个系统&#xff0c;因此浏览器和服务器可以通过安全连接进行通信。 这意味着在处理数据之前&#xff0c;已发送的数据已加密&#xff0c;传输并最终解密。 问题在于&…

WEB效能测试和负载测试部分截图

效能测试&#xff1a; 负载测试&#xff1a; 转载于:https://www.cnblogs.com/DOOM-scse/archive/2013/01/07/2849110.html