使用Spring Data Neo4j进行领域建模

大家好,威利在这里。 上次我告诉您,我正在使用Neo4j和Spring Data Neo4j构建Skybase CMDB,我很高兴收到很多对此的积极反馈。 我展示了一些代码,但没有那么多。 在本文中,我将向您展示如何使用Spring Data Neo4j在Skybase中构建人员配置项 (CI)。

人CI要求

我们将在这里简单地通过构建人员CI来开始。 出于各种原因而使人们进入CMDB很有用:它们使您可以定义细粒度的访问控制(例如,Jim可以将这样的应用程序部署到开发环境中; Eric可以将他想要的任何东西部署到他想要的任何地方;等等。 ); 它们使您可以定义将接收关键事件通知的组。 等等

我们的个人CI将有一个用户名,名字和姓氏,一些电话号码,一个电子邮件地址,一个经理,直接报告,最后是他或她工作的项目。 我们需要能够在列表视图中显示人员,在详细信息视图中显示给定人员,允许用户创建,编辑和删除人员等等。 例如,下面是列表视图的外观,至少现在是这样:



这是我们的详细信息视图的外观:



人与项目之间的关系具有关联的角色。 这种关系也是协作者列表的基础:如果至少有一个项目是他们两个成员,那么两个人就是协作者。

我们的简单要求应该足以显示编写Spring Data Neo4j代码的感觉。

创建人员和项目成员实体

首先,我们将创建人。 我已经取消了验证和JAXB批注,因为它们与我们当前的目的无关:

package org.skydingo.skybase.model;import java.util.Set;
import org.neo4j.graphdb.Direction;
import org.skydingo.skybase.model.relationship.ProjectMembership;
import org.springframework.data.neo4j.annotation.*;
import org.springframework.data.neo4j.support.index.IndexType;@NodeEntity
public class Person implements Comparable<Person> {@GraphId private Long id;@Indexed(indexType = IndexType.FULLTEXT, indexName = "searchByUsername")private String username;private String firstName, lastName, title, workPhone, mobilePhone, email;@RelatedTo(type = "REPORTS_TO")private Person manager;@RelatedTo(type = "REPORTS_TO", direction = Direction.INCOMING)private Set<Person> directReports;@RelatedToVia(type = "MEMBER_OF")private Set<ProjectMembership> memberships;public Long getId() { return id; }public void setId(Long id) { this.id = id; }public String getUsername() { return username; }public void setUsername(String username) { this.username = username; }... other accessor methods ...public Person getManager() { return manager; }public void setManager(Person manager) { this.manager = manager; }public Set<Person> getDirectReports() { return directReports; }public void setDirectReports(Set<Person> directReports) {this.directReports = directReports;}public Iterable<ProjectMembership> getMemberships() { return memberships; }public ProjectMembership memberOf(Project project, String role) {ProjectMembership membership = new ProjectMembership(this, project, role);memberships.add(membership);return membership;}... equals(), hashCode(), compareTo() ...
}

我们使用许多注释来放置结构。 让我们从节点及其属性开始。 然后,我们将研究节点之间的简单关系。 然后,我们看一下所谓的关系实体,它们基本上是奇特的关系。 首先,这是我们的域模型的抽象表示:

现在让我们看一些细节。

节点及其属性 。 当我们有一个节点支持的实体时,首先我们用@NodeEntity注释对其进行注释。 大多数简单的节点属性(即与其他节点没有关系的属性)都会随处可见。 请注意,我不必注释firstName,lastName,email等。 Spring Data Neo4j将在那里自动处理映射。

但是有两个例外。 第一个是我将@GraphId放在我的id属性上。 这告诉Spring Data Neo4j这是一个我们可以用于查找的标识符。 另一个是@Indexed注释,该注释(惊奇)为所讨论的属性创建了一个索引。 当您想要替代基于ID的查找时,这很有用。

现在我们来看一下关系。 概括地说,有简单的关系和更高级的关系。 我们将从简单的开始。

简单的关系 。 从低层次上来说,Neo4j是一个图形数据库,因此我们可以用图形理论术语来讨论图形,例如节点,边,有向边,DAG等。 但是这里我们使用图进行领域建模,因此我们根据更高层次的领域建模概念来解释低层次的图形概念。 Spring Data Neo4j使用的语言是“节点实体”代表节点,“关系”代表边缘。

我们的Person CI有一个简单的关系,称为REPORTS_TO,它与人们相关联,因此我们可以对报告层次结构进行建模。 Person有两个用于此关系的字段:manager和directReports。 这些是具有相同关系的相对站点。 我们使用@RelatedTo(type =“ REPORTS_TO”)注释这些字段。 注释还具有一个direction元素,其默认值为Direction.OUTGOING,这意味着“此”节点为边缘尾部。 这就是为什么我们为directReports字段明确指定direction = Direction.INCOMING的原因。

这在数据库中是什么样的? Neoclipse揭示了一切。 以下是一些报告关系示例(单击图像可查看大图):

(小写:有一个@Fetch注释-我们待会儿会看到它-告诉Spring Data Neo4j渴望加载相关实体。出于某种原因,我不必在管理器和直接报表关系中使用它,我不知道为什么。如果有人知道,我将不胜感激。

关系实体 。 除了人与人之间的REPORTS_TO关系之外,我们还关心人与项目之间的MEMBER_OF关系。 这比REPORTS_TO关系更有趣,因为MEMBER_OF具有关联的属性-角色-类似于向RDBMS中的链接表添加列,正如我在上一篇文章对Brig的答复中提到的那样。 Person.memberOf()方法提供了一种使用特殊ProjectMembership“关系实体”将人员分配到项目的便捷方法。 这是代码:

package org.skydingo.skybase.model.relationship;import org.skydingo.skybase.model.Person;
import org.skydingo.skybase.model.Project;
import org.springframework.data.neo4j.annotation.*;@RelationshipEntity(type = "MEMBER_OF")
public class ProjectMembership {@GraphId private Long id;@Fetch @StartNode private Person person;@Fetch @EndNode private Project project;private String role;public ProjectMembership() { }public ProjectMembership(Person person, Project project, String role) {this.person = person;this.project = project;this.role = role;}public Person getPerson() { return person; }public void setPerson(Person person) { this.person = person; }public Project getProject() { return project; }public void setProject(Project project) { this.project = project; }public String getRole() { return role; }public void setRole(String role) { this.role = role; }... equals(), hashCode(), toString() ...}

像Person一样,ProjectMembership是一个实体,但它是一个关系实体。 我们使用@RelationshipEntity(type =“ MEMBER_OF”)将其标记为关系实体,并且与Person一样,将@GraphId用作id属性。 @StartNode和@EndNode注释分别指示边缘的尾部和头部。 @Fetch告诉Spring Data Neo4j急切地加载节点。 默认情况下,Spring Data Neo4j不会急于加载关系,因为可能会将整个图形加载到内存中。

创建人仓库

这是我们的PersonRepository接口:

package org.skydingo.skybase.repository;import java.util.Set;
import org.skydingo.skybase.model.Person;
import org.skydingo.skybase.model.Project;
import org.springframework.data.neo4j.annotation.Query;
import org.springframework.data.neo4j.repository.GraphRepository;public interface PersonRepository extends GraphRepository<Person> {Person findByUsername(String username);@Query("start project=node({0}) match project<--person return person")Set<Person> findByProject(Project project);@Query("start person=node({0}) " +"match person-[:MEMBER_OF]->project<-[:MEMBER_OF]-collaborator " +"return collaborator")Set<Person> findCollaborators(Person person);
}

我在上一篇文章中指出,我们需要做的就是扩展GraphRepository接口。 Spring Data自动生成实现。

对于findByUsername(),Spring Data可以找出所需的查询。 对于其他两个查询,我们使用@Query和Cypher查询语言来指定所需的结果集。 查询中的{0}指的是finder方法参数。 在findCollaborators()查询中,我们使用[:MEMBER_OF]指示我们要遵循的关系。 这些返回Set而不是Iterables以消除重复项。

创建网页控制器

我们不会在这里介绍整个控制器,但是会介绍一些代表性的方法。 假设我们已经将一个PersonRepository注入到控制器中。

创建一个人 。 要创建一个人,我们可以使用以下方法:

@RequestMapping(value = "", method = RequestMethod.POST)
public String createPerson(Model model, @ModelAttribute Person person) {personRepo.save(person);return "redirect:/people?a=created";
}

再一次,我们忽略了验证。 我们要做的就是在存储库上调用save()方法。 这也是更新的工作方式。

寻找所有人 。 接下来,这是我们如何吸引所有人的方法:

@RequestMapping(value = "", method = RequestMethod.GET)
public String getPersonList(Model model) {Iterable<Person> personIt = personRepo.findAll();List<Person> people =new ArrayList<Person>(IteratorUtil.asCollection(personIt));Collections.sort(people);model.addAttribute(people);return "personList";
}

我们必须做一些工作才能使PersonRepository.findAll()返回的Iterable成为所需的格式。 Neo4j(org.neo4j.helpers.collection.IteratorUtil)随附的IteratorUtil在此提供帮助。

寻找一个人 。 在这里,我们要显示我们上面建立的个人详细信息。 与findAll()一样,我们必须做一些自我按摩:

@RequestMapping(value = "/{username}", method = RequestMethod.GET)
public String getPersonDetails(@PathVariable String username, Model model) {Person person = personRepo.findByUsername(username);List<ProjectMembership> memberships =CollectionsUtil.asList(person.getMemberships());List<Person> directReports =CollectionsUtil.asList(person.getDirectReports());List<Person> collaborators =CollectionsUtil.asList(personRepo.findCollaborators(person));Collections.sort(directReports);Collections.sort(collaborators);model.addAttribute(person);model.addAttribute("memberships", memberships);model.addAttribute("directReports", directReports);model.addAttribute("collaborators", collaborators);return "personDetails";
}

如果要查看JSP,请访问Skybase GitHub网站 。

配置APP

最后,这是我的beans-service.xml文件:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:context="http://www.springframework.org/schema/context"xmlns:neo4j="http://www.springframework.org/schema/data/neo4j"xmlns:p="http://www.springframework.org/schema/p"xmlns:tx="http://www.springframework.org/schema/tx"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans-3.0.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context-3.0.xsdhttp://www.springframework.org/schema/data/neo4jhttp://www.springframework.org/schema/data/neo4j/spring-neo4j-2.0.xsdhttp://www.springframework.org/schema/txhttp://www.springframework.org/schema/tx/spring-tx-3.0.xsd"><context:property-placeholderlocation="classpath:/spring/environment.properties" /><context:annotation-config /><context:component-scan base-package="org.skydingo.skybase.service" /><tx:annotation-driven mode="proxy" /><neo4j:config storeDirectory="${graphDb.dir}" /><neo4j:repositories base-package="org.skydingo.skybase.repository" />
</beans>

Neo4j具有一个基于POJO的基本映射模型和一个基于AspectJ的高级映射模型。 在此博客文章中,我们一直在使用基于POJO的基本方法,因此我们不需要包括与AspectJ相关的配置,例如<context:spring-configured>。

在那里,这就是Neo4j支持的Person CI。 编码愉快!

要更详细地查看代码或参与Skybase开发,请访问Skybase GitHub网站

参考:来自Skydingo博客的JCG合作伙伴 Willie Wheeler的Spring Data Neo4j进行领域建模 。

相关文章 :

  • Spring Data JPA的持久层
  • 基于事务的基于事件的NOSQL存储
  • SQL或NOSQL:这是问题吗?
  • 什么是NoSQL?
  • Cassandra,MongoDB,CouchDB,Redis,Riak,HBase比较

翻译自: https://www.javacodegeeks.com/2012/01/domain-modeling-with-spring-data-neo4j.html

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

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

相关文章

mysql 阿里内核人员

丁奇 http://dinglin.javaeye.com/鸣嵩 曹伟-鸣嵩 (新浪微博)彭立勋 http://www.penglixun.com/皓庭 http://wqtn22.iteye.com/项仲 http://blog.csdn.net/wudongxu剑川 http://gaoyusong.com/武藏 http://ybbct.iteye.com/祁奚 http://i.mtime.com/844165/褚霸 http://blog.y…

linux基本命令学习

1. 执行文件&#xff1a; ./文件 &#xff1a;执行该文件 sh startup.sh 执行该文件 2. Vi编辑 vi命令&#xff1a; :w 保存文件但不退出 :w file 将修改另外保存到file中&#xff0c;不退出 :w! 强制保存&#xff0c;不推出 :wq 保存文件并退出 :wq! 强制保存文件&#xff0…

506. 相对名次

方法一 复制数组排序返回对应下标 class Solution {public String[] findRelativeRanks(int[] score) {int n score.length;int[] newScore Arrays.copyOf(score,n);Arrays.sort(newScore);String[] res new String[n];for(int i 0;i<n;i){int rank Arrays.binarySear…

Java压缩

在最近的项目中&#xff0c;我们不得不做一些我个人从未真正看过的事情。 压缩。 我们需要拍几个文件和图像&#xff0c;将它们压缩并提供给FTP使用&#xff0c;是的&#xff0c;总有一天&#xff0c;感觉确实回到了90年代。 除了过去的FTP之行外&#xff0c;它还是一个很好的机…

苏州面对公司发布

假设您对我们这种创业型公司和我们的发展方向感兴趣的话&#xff0c;我们希望通过以下10个问答进一步添加两方的了解。我们希望看到的是您经过深思熟虑的、对公司和自己的前途负责任的谨慎回答。而不是应付公差式的轻描淡写&#xff08;我们会依据您回答质量的高低决定是否邀请…

linux多线程_Java+Linux,深入内核源码讲解多线程之进程

之前写了两篇文章&#xff0c;都是针对Linux这个系统的&#xff0c;为什么?我为什么这么喜欢写这个系统的知识&#xff0c;可能就是为了今天的内容多线程系列&#xff0c;现在多线程不是一个面试重点 啊&#xff0c;那如果你能深入系统内核回答这个知识点&#xff0c;面试官会…

594. 最长和谐子序列

和谐数组是指一个数组里元素的最大值和最小值之间的差别 正好是 1 。 现在&#xff0c;给你一个整数数组 nums &#xff0c;请你在所有可能的子序列中找到最长的和谐子序列的长度。 数组的子序列是一个由数组派生出来的序列&#xff0c;它可以通过删除一些元素或不删除元素、…

解决git clone报错SSL certificate problem

Git新手一枚&#xff0c;今天进行git clone操作时发生如下问题&#xff1a;提示无效的链接error: SSL certificate problem: Invalid certificate chain while accessing https://githib.com/...XXXX.git fatal: HTTP request failed解决方法也很简单&#xff0c;一条命令就搞定…

使用内存映射文件获取巨大的矩阵

总览 矩阵可能真的很大&#xff0c;有时甚至比一个数组中可以容纳的更大。 您可以通过具有多个数组来扩展最大大小&#xff0c;但这会使堆大小确实很大且效率低下。 一种替代方法是在内存映射文件上使用包装器。 内存映射文件的优点是它们对堆的影响很小&#xff0c;并且可以由…

ipad连接电脑_这些应用让iPad生产力分分钟UP

IT时报见习记者 钱奕昀用iPad办公这件事&#xff0c;多年前网友就在讨论&#xff0c;最常见的还是那句“买前生产力&#xff0c;买后爱奇艺”。很长一段时间里&#xff0c;它的生产力属性都是弱于娱乐属性的。其实&#xff0c;作为PC端和移动端的形态中和&#xff0c;iPad可以…

Mac OSX 快捷键命令行

ctrlshift 快速放大dock的图标会暂时放大&#xff0c;而如果你开启了dock放大CommandOptionW 将所有窗口关闭CommandW 将当前窗口关闭(可以关闭Safari标签栏,很实用) CommandOptionM …

将JavaFX 2.0与Swing和SWT集成

JavaFX 2.0对JavaFX的改进之一是可以更轻松地与Swing和SWT进行互操作 。 一些在线资源记录了如何完成此操作。 其中包括将JavaFX集成到Swing应用程序和SWT Interop中 。 但是&#xff0c;在有效的类级Javadoc文档的一个很好的示例中&#xff0c;各自的JavaFX类javafx.embed.swi…

iOS-如何返回某个字符串的拼音助记码

我也是看了网上的一个示例代码后&#xff0c;在它的基础上进行的修改。因为项目上会用到&#xff0c;我相信很多人的项目上也会用到。所以实现后&#xff0c;也赶紧分享出来&#xff0c;希望后来人不需要花费时间了。 提示&#xff1a;这里用到了正则表达式&#xff0c;使用了一…

wifi rssi 计算 距离_WiFi和WLAN是一样的?真相在这里~别再傻傻分不清了

我们通常上网的时候会说连接WiFi如果注意到无线网络的名称就会发现手机的连接显示是WLAN别再将WiFI和WLAN搞混了&#xff01;二者的定义WLANWLAN的全称为 Wireless Local Area Networks,中文意思为无线局域网络&#xff0c;是一种数据传输系统。它是利用射频技术进行数据传输&a…

【Shell剧本练习】得出的结论是当前用户

推断是否当前用户root。假设是暗示root用户&#xff0c;假设而不是提示对于普通用户#!/bin/bash #title: testus.sh #author: orangleliu #date: 2014-08-09 #desc: get current user, if it is root user, tell us it is super user or tell us is a common user# #Function C…

播放框架模块:分而治之

通常情况是您开始开发应用程序并继续满足要求。 当您的应用程序变得更大时&#xff0c;您开始意识到将其分为不同组件的便利。 而且&#xff0c;当您开发第二个或第三个应用程序时&#xff0c;您开始认识到可以在不同应用程序之间重用的某些功能。 这是模块化应用程序的两个很好…

Alpha阶段项目总结

1.我们的软件要解决什么问题&#xff1f;是否定义得很清楚&#xff1f;是否对典型用户和典型场景有清晰的描述&#xff1f; 我们的软件是一款针对健康饮食而做的一款饮食健康软件&#xff0c;对生活中我们经常迟到的很多事物组合都进行了详细的注解&#xff0c;用户可以清楚地看…

实用的it知识学习_怎样能更快更好的学习好书法?分享一些比较实用的理论知识...

如何能更快更高效的学习书法&#xff1f;首先了解一些书法理论知识是很有必要的&#xff01;它能让你在学习书法的过程中不至于迷茫 &#xff01;能助你更快学好书法&#xff01;一、书论在实践中产生我们大部分人都觉得学习书法可以没有理论&#xff0c;但不可无技法。但理论和…

九度oj-1001-Java

题目描述&#xff1a; This time, you are supposed to find AB where A and B are two matrices, and then count the number of zero rows and columns. 输入&#xff1a; The input consists of several test cases, each starts with a pair of positive integers M and N …

字节流与字符流的区别

最近在项目中遇到一个encoding的问题&#xff0c;记录一下。 具体而言就是&#xff0c;项目中有A/B两个部分&#xff0c;A部分由我们负责&#xff0c;Java实现&#xff1b;B部分是UK负责的&#xff0c;使用Delphi&#xff0c;A、B在交互时发送一个http请求&#xff0c; 请求汇总…