neo4j遍历和图算法
这篇blog主要和大家分享一下neo4j中是如何对节点进行遍历,和其中集成的图论的一些常用算法。
遍历
http://docs.neo4j.org.cn/tutorials-java-embedded-traversal.html 这是neo4j官方的中文教程。
- private static Traverser getFriends(Node node )
- {
- TraversalDescription td = Traversal.description()
- .breadthFirst()
- .relationships( RelTypes.KNOWS, Direction.OUTGOING )
- .evaluator( Evaluators.excludeStartPosition() );
- return td.traverse( node );
- }
private static Traverser getFriends(Node node )
{
TraversalDescription td = Traversal.description()
.breadthFirst()
.relationships( RelTypes.KNOWS, Direction.OUTGOING )
.evaluator( Evaluators.excludeStartPosition() );
return td.traverse( node );
}
TraversalDescription提供了用户遍历节点的方法,并且用户需要自己添加遍历的条件。
- int num = 0;
- String result = neoNode.getProperty( "name" ) + "'s friends:\n";
- Traverser friendsTraverser = getFriends( neoNode );
- for ( Path friendPath : friendsTraverser )
- {
- output += "At depth " + friendPath.length() + " => "
- + friendPath.endNode()
- .getProperty( "name" ) + "\n";
- numberOfFriends++;
- }
- output += "Number of friends found: " + numberOfFriends + "\n";
int num = 0;
String result = neoNode.getProperty( "name" ) + "'s friends:\n";
Traverser friendsTraverser = getFriends( neoNode );
for ( Path friendPath : friendsTraverser )
{
output += "At depth " + friendPath.length() + " => "
+ friendPath.endNode()
.getProperty( "name" ) + "\n";
numberOfFriends++;
}
output += "Number of friends found: " + numberOfFriends + "\n";
打印出结果
- Thomas Anderson's friends:
- At depth 1 => Trinity
- At depth 1 => Morpheus
- At depth 2 => Cypher
- At depth 3 => Agent Smith
- 找到朋友的数量: 4
Thomas Anderson's friends:
At depth 1 => Trinity
At depth 1 => Morpheus
At depth 2 => Cypher
At depth 3 => Agent Smith
找到朋友的数量: 4
下面介绍如何按照一个预先定义好的顺序遍历图。
- Node A = db.createNode();
- Node B = db.createNode();
- Node C = db.createNode();
- Node D = db.createNode();
- A.createRelationshipTo( B, REL1 );
- B.createRelationshipTo( C, REL2 );
- C.createRelationshipTo( D, REL3 );
- A.createRelationshipTo( C, REL2 );
Node A = db.createNode();
Node B = db.createNode();
Node C = db.createNode();
Node D = db.createNode();
A.createRelationshipTo( B, REL1 );
B.createRelationshipTo( C, REL2 );
C.createRelationshipTo( D, REL3 );
A.createRelationshipTo( C, REL2 );
现在关系(REL1-->REL2-->REL), 当遍历的时候,Evaluator能够对它进行检验,确保只有该关系顺序的路径才会被包括。
- ArrayList<RelationshipType> orderedPathContext = new ArrayList<RelationshipType>();
- orderedPathContext.add( REL1 );
- orderedPathContext.add( withName( "REL2" ) );
- orderedPathContext.add( withName( "REL3" ) );
- TraversalDescription td = Traversal.description()
- .evaluator( new Evaluator()
- {
- @Override
- public Evaluation evaluate( final Path path )
- {
- if ( path.length() == 0 )
- {
- return Evaluation.EXCLUDE_AND_CONTINUE;
- }
- RelationshipType expectedType = orderedPathContext.get( path.length() - 1 );
- boolean isExpectedType = path.lastRelationship()
- .isType( expectedType );
- boolean included = path.length() == orderedPathContext.size()
- && isExpectedType;
- boolean continued = path.length() < orderedPathContext.size()
- && isExpectedType;
- return Evaluation.of( included, continued );
- }
- } );
ArrayList<RelationshipType> orderedPathContext = new ArrayList<RelationshipType>();
orderedPathContext.add( REL1 );
orderedPathContext.add( withName( "REL2" ) );
orderedPathContext.add( withName( "REL3" ) );
TraversalDescription td = Traversal.description()
.evaluator( new Evaluator()
{
@Override
public Evaluation evaluate( final Path path )
{
if ( path.length() == 0 )
{
return Evaluation.EXCLUDE_AND_CONTINUE;
}
RelationshipType expectedType = orderedPathContext.get( path.length() - 1 );
boolean isExpectedType = path.lastRelationship()
.isType( expectedType );
boolean included = path.length() == orderedPathContext.size()
&& isExpectedType;
boolean continued = path.length() < orderedPathContext.size()
&& isExpectedType;
return Evaluation.of( included, continued );
}
} );
打印出结果
- Traverser traverser = td.traverse( A );
- PathPrinter pathPrinter = new PathPrinter( "name" );
- for ( Path path : traverser )
- {
- output += Traversal.pathToString( path, pathPrinter );
- }
Traverser traverser = td.traverse( A );
PathPrinter pathPrinter = new PathPrinter( "name" );
for ( Path path : traverser )
{
output += Traversal.pathToString( path, pathPrinter );
}
用户也可以通过自定义格式输出路径。
- static class PathPrinter implements Traversal.PathDescriptor<Path>
- {
- private final String nodePropertyKey;
- public PathPrinter( String nodePropertyKey )
- {
- this.nodePropertyKey = nodePropertyKey;
- }
- @Override
- public String nodeRepresentation( Path path, Node node )
- {
- return "(" + node.getProperty( nodePropertyKey, "" ) + ")";
- }
- @Override
- public String relationshipRepresentation( Path path, Node from,
- Relationship relationship )
- {
- String prefix = "--", suffix = "--";
- if ( from.equals( relationship.getEndNode() ) )
- {
- prefix = "<--";
- }
- else
- {
- suffix = "-->";
- }
- return prefix + "[" + relationship.getType().name() + "]" + suffix;
- }
- }
static class PathPrinter implements Traversal.PathDescriptor<Path>
{
private final String nodePropertyKey;public PathPrinter( String nodePropertyKey )
{
this.nodePropertyKey = nodePropertyKey;
} @Override
public String nodeRepresentation( Path path, Node node )
{
return "(" + node.getProperty( nodePropertyKey, "" ) + ")";
} @Override
public String relationshipRepresentation( Path path, Node from,
Relationship relationship )
{
String prefix = "--", suffix = "--";
if ( from.equals( relationship.getEndNode() ) )
{
prefix = "<--";
}
else
{
suffix = "-->";
}
return prefix + "[" + relationship.getType().name() + "]" + suffix;
}
}
图算法
- public Iterable<Path> findShortestPath(Node node1, Node node2) {
- PathFinder<Path> finder = GraphAlgoFactory.shortestPath(
- Traversal.expanderForTypes(RelTypes.KNOWS, Direction.BOTH), 10);
- Iterable<Path> paths = finder.findAllPaths(node1, node2);
- return paths;
- }
public Iterable<Path> findShortestPath(Node node1, Node node2) {
PathFinder<Path> finder = GraphAlgoFactory.shortestPath(
Traversal.expanderForTypes(RelTypes.KNOWS, Direction.BOTH), 10);
Iterable<Path> paths = finder.findAllPaths(node1, node2);
return paths;
}
- for(Path shortestPath: findShortestPath(aa, ab)) {
- System.out.println(shortestPath.toString());
- }
for(Path shortestPath: findShortestPath(aa, ab)) {
System.out.println(shortestPath.toString());
}
Traversal.expanderForTypes用来定义遍历的深度和节点关系的出入度。
- PathFinder<WeightedPath> finder = GraphAlgoFactory.dijkstra(
- Traversal.expanderForTypes( ExampleTypes.MY_TYPE, Direction.BOTH ), "cost" );
- WeightedPath path = finder.findSinglePath( nodeA, nodeB );
- path.weight();
PathFinder<WeightedPath> finder = GraphAlgoFactory.dijkstra(
Traversal.expanderForTypes( ExampleTypes.MY_TYPE, Direction.BOTH ), "cost" ); WeightedPath path = finder.findSinglePath( nodeA, nodeB );
path.weight();
使用 A*算法是解决静态路网中求解最短路最有效的方法。
- Node nodeA = createNode( "name", "A", "x", 0d, "y", 0d );
- Node nodeB = createNode( "name", "B", "x", 7d, "y", 0d );
- Node nodeC = createNode( "name", "C", "x", 2d, "y", 1d );
- Relationship relAB = createRelationship( nodeA, nodeC, "length", 2d );
- Relationship relBC = createRelationship( nodeC, nodeB, "length", 3d );
- Relationship relAC = createRelationship( nodeA, nodeB, "length", 10d );
- EstimateEvaluator<Double> estimateEvaluator = new EstimateEvaluator<Double>()
- {
- public Double getCost( final Node node, final Node goal )
- {
- double dx = (Double) node.getProperty( "x" ) - (Double) goal.getProperty( "x" );
- double dy = (Double) node.getProperty( "y" ) - (Double) goal.getProperty( "y" );
- double result = Math.sqrt( Math.pow( dx, 2 ) + Math.pow( dy, 2 ) );
- return result;
- }
- };
- PathFinder<WeightedPath> astar = GraphAlgoFactory.aStar(
- Traversal.expanderForAllTypes(),
- CommonEvaluators.doubleCostEvaluator( "length" ), estimateEvaluator );
- WeightedPath path = astar.findSinglePath( nodeA, nodeB );
Node nodeA = createNode( "name", "A", "x", 0d, "y", 0d );
Node nodeB = createNode( "name", "B", "x", 7d, "y", 0d );
Node nodeC = createNode( "name", "C", "x", 2d, "y", 1d );
Relationship relAB = createRelationship( nodeA, nodeC, "length", 2d );
Relationship relBC = createRelationship( nodeC, nodeB, "length", 3d );
Relationship relAC = createRelationship( nodeA, nodeB, "length", 10d ); EstimateEvaluator<Double> estimateEvaluator = new EstimateEvaluator<Double>()
{
public Double getCost( final Node node, final Node goal )
{
double dx = (Double) node.getProperty( "x" ) - (Double) goal.getProperty( "x" );
double dy = (Double) node.getProperty( "y" ) - (Double) goal.getProperty( "y" );
double result = Math.sqrt( Math.pow( dx, 2 ) + Math.pow( dy, 2 ) );
return result;
}
};
PathFinder<WeightedPath> astar = GraphAlgoFactory.aStar(
Traversal.expanderForAllTypes(),
CommonEvaluators.doubleCostEvaluator( "length" ), estimateEvaluator );
WeightedPath path = astar.findSinglePath( nodeA, nodeB );
neo4j遍历和图算法的更多相关文章
- 关于图计算&图学习的基础知识概览:前置知识点学习(Paddle Graph Learning (PGL))
关于图计算&图学习的基础知识概览:前置知识点学习(Paddle Graph Learning (PGL)) 欢迎fork本项目原始链接:关于图计算&图学习的基础知识概览:前置知识点学习 ...
- 20172305 2018-2019-1 《Java软件结构与数据结构》第九周学习总结
20172305 2018-2019-1 <Java软件结构与数据结构>第九周学习总结 教材学习内容总结 本周内容主要为书第十五章内容: 图(结点和结点之间的连接构成) 顶点:结点 边:结 ...
- CF-544:部分题目总结
-------------------昨天打的重现赛,感觉是我打的发挥的最好的一场比赛了,六题都一次AC.那么就来总结一下吧 题目链接:http://codeforces.com/contest/11 ...
- Neo4j简介
Neo4j简介 发表于2013年3月16日 11:52 p.m. 位于分类图数据库与图并行计算 现实中很多数据都是用图来表达的,比如社交网络中人与人的关系.地图数据.或是基因信息等等.RDBMS ...
- 为什么选择图形数据库,为什么选择Neo4j?
最近在抓取一些社交网站的数据,抓下来的数据用MySql存储.问我为什么用MySql,那自然是入门简单,并且我当时只熟悉MySql.可是,随着数据量越来越大,有一个问题始终困扰着我,那就是社交关系的存储 ...
- 主流图数据库Neo4J、ArangoDB、OrientDB综合对比:架构分析
主流图数据库Neo4J.ArangoDB.OrientDB综合对比:架构分析 YOTOY 关注 0.4 2017.06.15 15:11* 字数 3733 阅读 16430评论 2喜欢 18 1: 本 ...
- Neo4j
Neo4j是一个高性能的,NOSQL图形数据库,它将结构化数据存储在网络上而不是表中.它是一个嵌入式的.基于磁盘的.具备完全的事务特性的Java持久化引擎,但是它将结构化数据存储在网络(从数学角度叫做 ...
- 图数据库neo4j和关系数据库的区别
相信您和我一样,在使用关系型数据库时常常会遇到一系列非常复杂的设计问题.例如一部电影中的各个演员常常有主角配角之分,还要有导演,特效等人员的参与.通常情况下这些人员常常都被抽象为Person类型,对应 ...
- Neo4j图数据库管理系统开发笔记之一:Neo4j Java 工具包
1 应用开发概述 基于数据传输效率以及接口自定义等特殊性需求,我们暂时放弃使用Neo4j服务器版本,而是在Neo4j嵌入式版本的基础上进行一些封装性的开发.封装的重点,是解决Neo4j嵌入式版本Emb ...
随机推荐
- AtCoder F - Exhausted?
传送门 sxy题解: //Achen #include<algorithm> #include<iostream> #include<cstring> #inclu ...
- koa2 安装环境
1.安装koa2 npm install koa@2 -g 2.创建项目 (1)安装koa2生成器 npm install koa-generator -g (2)koa2生成一个test项目 koa ...
- C++ 字符串、string、char *、char[]、const char*的转换和区别
1.字符串 字符串本质就是一串字符,在C++中大家想到字符串往往第一反应是std::string(后面简称string) 字符串得从C语言说起,string其实是个类,C语言是没有class的,所以C ...
- java基础之Character类概述
Character 类 在对象中包装一个基本类型 char 的值 此外,该类提供了几种方法,以确定字符的类别(小写字母,数字,等等),并将字符从大写转换成小写,反之亦然 构造方法 public Cha ...
- Django之模板语言(二)-----Filter
1.其他常用的模板语言: 通过模板语言可以让前端页面显示数据,数据可以是基本数据类型,也可以是对象亦或者对象的列表,结合着模板中的for.if等配合使用. 要注意前端页面中,出现没有后端数据的情况,随 ...
- How to ping and test for a specific port from Linux or Unix command line
Use telnet command The syntax is:telnet {host} {port}telnet www.cyberciti.biz 80telnet 192.168.2.254 ...
- 网络结构解读之inception系列四:Inception V3
网络结构解读之inception系列四:Inception V3 Inception V3根据前面两篇结构的经验和新设计的结构的实验,总结了一套可借鉴的网络结构设计的原则.理解这些原则的背后隐藏的 ...
- 查找父进程,进程的PEB 进程是否被调试 NtQueryInformationProcess
这个函数的功能很强大,可以用来查找进程的很多相关信息. 先看一下定义: NTSTATUS WINAPI NtQueryInformationProcess( _In_ HANDLE ProcessHa ...
- HZOI2019建造游乐园(play)组合数学,欧拉图
题目:https://www.cnblogs.com/Juve/articles/11186805.html(密码是我的一个oj用户名) solution: 反正我是想不出来... 题目大意就是要求出 ...
- 强制以32位ie运行程序
最近被一个问题给郁闷住了.给电脑重装系统后,发现发布好的程序.或者VS2012总是以64位ie运行程序,这样的话skyline的三维控件无法显示.到现在我是确定ie64无法识别skyline的控件. ...