Neo4j

第一部分 图 和 Neo4j 1.1 图论 1.1.1 图论起源 --- 柯尼斯堡( Konigsberg )七桥问题 众所周知,图论起源于一个非常经典的问题 —— 柯尼斯堡( Konigsberg )七桥问题。 1738 年,瑞典数 学家欧拉 ( Leornhard Euler) 解决了柯尼斯堡七桥问题。由此图论诞生,欧拉也成为图论的创始人。 欧拉把问题的实质归于 " 一笔画 " 问题,即判断一个图是否能够遍历完所有的边 (Edge) 而没有重复,而柯 尼斯堡七桥问题则是一笔画问题的一个具体情境。欧拉证明这个问题不成立。 满足一笔画的图满足两个条件 : 图必须是一个完整图 有零个或二个奇数点 1.1.2 图 和 节点 图是一组节点和连接这些节点的关系组成。图形数据存储在节点和关系所在的属性上。属性是键值对表 示的数据。 在图形理论中,我们可以使用圆表示一个节点 并且可以向里面添加键值对形式的数据。 1.1.3 节点关系表达 简单关系表达 此处在两个节点之间创建关系名称 “ 跟随 ” 。 这意味着 Profifile1 跟随 Profifile2 。 复杂关系表达 这里节点用关系连接。 关系是单向或双向的。 从 XYZ 到 PQR 的关系是单向关系。 从 ABC 到 XYZ 的关系是双向关系。 1.1.4 属性图模型规则 图表示节点,关系和属性中的数据 节点和关系都包含属性 关系连接节点 属性是键值对 节点用圆圈表示,关系用方向键表示。 关系具有方向:单向和双向。 每个关系包含 “ 开始节点 ” 或 “ 从节点 ” 和 “ 到节点 ” 或 “ 结束节点 ” 1.2 知识图谱和图库 1.2.1 知识图谱 一种基于图的数据结构,由节点 (Node) 和边 (Edge) 组成。其中节点即实体,由一个全局唯一的 ID 标示, 边就是关系用于连接两个节点。通俗地讲,知识图谱就是把所有不同种类的信息( Heterogeneous Information )连接在一起而得到的一个关系网络。知识图谱提供了从 “ 关系 ” 的角度去分析问题的能力。 互联网、大数据的背景下,谷歌、百度、搜狗等搜索引擎纷纷基于该背景,创建自己的知识图谱 Knowledge Graph (谷歌)、知心(百度)和知立方(搜狗),主要用于改进搜索质量。 1.2.2 图数据库 一般情况下,我们使用数据库查找事物间的联系的时候,只需要短程关系的查询(两层以内的关联)。 当需要进行更长程的,更广范围的关系查询时,就需要图数据库的功能。而随着社交、电商、金融、零 售、物联网等行业的快速发展,现实世界的事物之间织起了一张巨大复杂的关系网,传统数据库面对这 样复杂关系往往束手无策。因此,图数据库应运而生。 图数据库 (Graph database) 指的是以图数据结构的形式来存储和查询数据的数据库。 知识图谱中,知识的组织形式采用的就是图结构,所以非常适合用图库进行存储。 1.2.3 图形数据库优势 在需要表示多对多关系时,我们常常需要创建一个关联表来记录不同实体的多对多关系。如果两个实体 之间拥有多种关系,那么我们就需要在它们之间创建多个关联表。而在一个图形数据库中,我们只需要 标明两者之间存在着不同的关系。如果希望在两个结点集间建立双向关系,我们就需要为每个方向定义 一个关系。 也就是说,相对于关系型数据库中的各种关联表,图形数据库中的关系可以通过关系属性这 一功能来提供更为丰富的关系展现方式。因此相较于关系型数据库,图形数据库的用户在对现实进行抽 象时将拥有一个额外的武器,那就是丰富的关系。 优势总结 : 性能上,对长程关系的查询速度快 擅于发现隐藏的关系,例如通过判断图上两点之间有没有走的通的路径,就可以发现事物间的关联 1.3 Neo4j 基础 1.3.1 什么是 Neo4j Neo4j 是一个开源的 无 Shcema 的 基于 java 开发的 图形数据库,它将结构化数据存储在图中而不 是表中。它是一个嵌入式的、基于磁盘的、具备完全的事务特性的 Java 持久化引擎。程序数据是 在一个面向对象的、灵活的网络结构下,而不是严格、静态的表中 , 但可以享受到具备完全的事务 特性、企业级的数据库的所有好处。 https://db-engines.com/en/ranking 1.3.2 Neo4j 模块构建 Neo4j 主要构建块 节点 属性 关系 标签 数据浏览器 节点 节点是图表的基本单位。 它包含具有键值对的属性 属性 属性是用于描述图节点和关系的键值对 Key = 值 其中 Key 是一个字符串 值可以通过使用任何 Neo4j 数据类型来表示 关系 关系是图形数据库的另一个主要构建块。 它连接两个节点,如下所示。 这里 Emp 和 Dept 是两个不同的节点。 “WORKS_FOR” 是 Emp 和 Dept 节点之间的关系。 因为它表示从 Emp 到 Dept 的箭头标记,那么这种关系描述的一样 Emp WORKS_FOR Dept 每个关系包含一个起始节点和一个结束节点。 这里 “Emp” 是一个起始节点。 “Dept” 是端节点。 由于该关系箭头标记表示从 “Emp” 节点到 “Dept” 节点的关系,该关系被称为 “ 进入关系 ” 到 “Dept” 节点。 并且 “ 外向关系 ” 到 “Emp” 节点。 像节点一样,关系也可以包含属性作为键值对。 标签 Label 将一个公共名称与一组节点或关系相关联。 节点或关系可以包含一个或多个标签。 我们可以为现 有节点或关系创建新标签。 我们可以从现有节点或关系中删除现有标签。 从前面的图中,我们可以观察到有两个节点。 左侧节点都有一个标签: “EMP” ,而右侧节点都有一个标签: “Dept” 。 这两个节点之间的关系,也有一个标签: “WORKS_FOR” 注: - Neo4j 将数据存储在节点或关系的属性中。 1.3.3 Neo4j 的主要应用场景 社交媒体和社交网络 当使用图形数据库为社交网络应用程序提供动力时,可以轻松利用社交关系或根据活动推断关系。 查询社区聚类分析,朋友的朋友推荐,影响者分析,共享和协作关系分析等 推荐引擎和产品推荐系统 图形驱动的推荐引擎通过实时利用多种连接,帮助公司个性化产品,内容和服务。 内容和媒体推荐,图形辅助搜索引擎,产品推荐,专业网络,社会推荐。 身份和访问管理 使用图形数据库进行身份和访问管理时,可以快速有效地跟踪用户,资产,关系和授权。 查询访问管理,资产来源,数据所有权,身份管理,互连组织,主数据,资源授权 金融反欺诈多维关联分析场景 通过图分析可以清楚地知道洗钱网络及相关嫌疑,例如对用户所使用的帐号、发生交易时的 IP 地址、 MAC 地址、手机 IMEI 号等进行关联分析。 1.3.4 Neo4j 环境搭建 Neo4j 环境 Linux 下搭建 (1). 切换到 Linux 下 到安装目录 neo4j 上传安装包 或者 下载安装包 使用 ftp 工具上传 neo4j-community-3.5.17.tar 到 liunx 下 或者 wget https://neo4j.com/artifact.php?name=neo4j-community-3.5.17-unix.tar.gz (2). 解压 tar -xvf neo4j-community-3.5.17.tar (3). 修改配置文件 neo4j.conf vi conf/neo4j.conf 主要是修改 允许远程访问的地址 把对应的注释打开即可 dbms.connectors.default_listen_address=0.0.0.0 (4). 开放对应的访问端口 默认要开放 7474 和 7687 fifirewall-cmd --zone=public --add-port=7474/tcp --permanent fifirewall-cmd --zone=public --add-port=7687/tcp --permanent systemctl reload fifirewalld (5). 启动 ./bin/neo4j start (6). 使用浏览器 访问服务器上的 neo4j http://192.168.211.133:7474 默认的账号是 neo4j 密码 neo4j 这里第一次登录的话会要求修改密码 Windows 下的安装 (1). 从 https://neo4j.com/download-center/#community 下载最新的 Neo4j Server 安装文件 可以看到 neo4J 软件 exe 或 zip 格式的所有版本 (2). 下载 Neo4j 3.5.17 (zip) (3). 解压 (4). 修改配置文件 dbms.connectors.default_listen_address=0.0.0.0 (5) . 通过 neo4j.bat install-service 安装 neo4j 服务 注意的问题 如果是 4.0 以及以上版本需要 jdk11 修改文件 bin/neo4j.ps1 Import-Module "neo4j 的主目录 \bin\Neo4j-Management.psd1" (6).neo4j.bat 启动 neo4j.bat start (7). 使用浏览器 访问服务器上的 neo4j http://127.0.0.1:7474 默认的账号是 neo4j 密码 neo4j 这里第一次登录的话会要求修改密码 1.3.5 Neo4j 数据浏览器 数据浏览器访问 一旦我们安装 Neo4j ,我们可以访问 Neo4j 数据浏览器使用以下 URL http:// localhost:7474/browser/ Neo4j 数据浏览器用于执行 CQL 命令并查看输出输出。 这里我们需要在美元提示符处执行所有 CQL 命令: “$” 如 CREATE(cc:CreditCard) 在美元符号后键入命令,然后单击 “ 执行 ” 按钮运行命令。 它与 Neo4j 数据库服务器交互,检索和显示下面的结果到那个美元提示。 使用 “VI 视图 ” 按钮以图形格式查看结果。 上图以 “UI 视图 ” 格式显示结果。 导出 CSV 或者 JSON 单击 “ 导出 CSV” 按钮以 csv 文件格式导出结果 第二部分 Neo4j CQL 2.1 CQL 简介 CQL 代表 Cypher 查询语言。 像关系型数据库具有查询语言 SQL , Neo4j 使用 CQL 作为查询语言。 Neo4j CQL 它是 Neo4j 图形数据库的查询语言。 它是一种声明性模式匹配语言。 它遵循 SQL 语法。 它的语法是非常简单且人性化、可读的格式。 常用的 Neo4j CQL 命令 / 条款如下: S.No. CQL 命令 / 作用 1 CREATE 创建 创建节点,关系和属性 2 MATCH 匹配 检索有关节点,关系和属性数据 3 RETURN 返回 返回查询结果 4 WHERE 条件 提供条件过滤检索 MATCH 数据 5 DELETE 删除 删除节点和关系 6 REMOVE 移除 删除节点和关系的属性 7 SET 设置 添加或更新标签 8 ORDER BY 排序 对结果排序 9 SKIP LIMIT 分页 分页 10 DISTINCT 排重 排重 完成上面的语法 我们基于庆余年电视剧人物关系图片 : 2.2 CREATE 语法元素 描述 < node-name> 它是我们将要创建的节点名称。 < label-name> 它是一个节点标签名称 < property1-name>...< propertyn name> 属性是键值对。 定义将分配给创建节点的属性的名 称 < property1-value>...< propertyn value> 属性是键值对。 定义将分配给创建节点的属性的值 语法说明: 举例 : 2.3 MATCH RETURN 命令语法 CREATE ( <node-name>:<label-name> [{ <property1-name>:<property1-Value> ........ <propertyn-name>:<propertyn-Value> }] ) CREATE (person:Person) CREATE (person:Person {cid:1,name:" 范 闲 ",age:24,gender:0,character:"A",money:1000}); CREATE (person:Person {cid:2,name:" 林婉 儿 ",age:20,gender:1,character:"B",money:800}); CREATE (person:Person {cid:3,name:" 庆 帝 ",age:49,gender:0,character:"A",money:8900}); MATCH ( <node-name>:<label-name> ) RETURN <node-name>.<property1-name>, ... <node-name>.<propertyn-name> 语法元素 描述 < node-name> 它是我们将要创建的节点名称。 < label-name> 它是一个节点标签名称 < property1-name>...< propertyn name> 属性是键值对。 定义将分配给创建节点的属性的名 称 举例 : 2.4 关系创建 S.No. 语法元素 描述 1 MATCH,CREATE,RETURN 他们是 Neo4J CQL 关键字。 2 < noode1-name> 它用于创建关系的 “From Node” 的名称。 3 < node1-label-name> 它用于创建关系的 “From Node” 的标签名称。 4 < node2-name> 它用于创建关系的 “To Node” 的名称。 5 < node2-label-name> 它用于创建关系的 “To Node” 的标签名称。 6 < relationship-name> 这是一个关系的名称。 7 < relationship-label-name> 它是一个关系的标签名称。 使用现有节点创建没有属性的关系 语法说明 : 使用现有节点创建有属性的关系 MATCH (person:Person) return person MATCH (person:Person) return person.name,person.age MATCH (<node1-name>:<node1-label-name>),(<node2-name>:<node2-label-name>) CREATE (<node1-name>)-[<relationship-name>:<relationship-label-name>]->(<node2- name>) RETURN 相应的内容 创建关系 match(person:Person {name:" 范闲 "}) ,(person2:Person {name:" 林婉儿 "}) create(person)-[r:Couple]->(person2); 查询关系 match p = (person:Person {name:" 范闲 "})-[r:Couple]->(person2:Person) return p match (p1:Person {name:" 范闲 "})-[r:Couple]-(p2:Person) return p1,p2 match (p1:Person {name:" 范闲 "})-[r:Couple]-(p2:Person) return r MATCH (<node1-label-name>:<node1-name>),(<node2-label-name>:<node2-name>) CREATE (<node1-label-name>)-[<relationship-label-name>:<relationship-name> {<define-properties-list>}]->(<node2-label-name>) RETURN <relationship-label-name> 其中 <define-properties-list> 是分配给新创建关系的属性(名称 - 值对)的列表。 { <property1-name>:<property1-value>, <property2-name>:<property2-value>, ... <propertyn-name>:<propertyn-value> } match(person:Person {name:" 范闲 "}),(person2:Person {name:" 林婉儿 "}) create (person)-[r:Couple{mary_date:"12/12/2014",price:55000}]->(person2) return r; 使用新节点创建没有属性的关系 CREATE (<node1-label-name>:<node1-name>) -[<relationship-label-name>:<relationship-name>]-> (<node1-label-name>:<node1-name>) create(person1:Person {cid:4,name:" 长公 主 ",age:49,gender:1,character:"A",money:5000}) -[r:Friend]-> (person2:Person {cid:7,name:" 九品射手燕小 乙 ",age:48,gender:0,character:"B",money:1000}) 使用新节点创建有属性的关系 CREATE (<node1-label-name>:<node1-name>{<define-properties-list>}) -[<relationship-label-name>:<relationship-name>{<define-properties-list>}] ->(<node1-label-name>:<node1-name>{<define-properties-list>}) create (person1:Person {cid:9,name:" 靖王世 子 ",age:23,gender:0,character:"A",money:3000}) <-[r:Friend {date:"11-02-2000"}]-> (person2:Person {cid:8,name:" 二皇子 ",age:24,gender:0,character:"B",money:6000}) 关系和节点的属性可以使用的类型 Index CQL 数据类型 作用 1. boolean 它用于表示布尔文字: true, false 。 2. byte 它用来表示 8 位整数。 3. short 它用于表示 16 位整数。 4. int 它用于表示 32 位整数。 5. long 它用于表示 64 位整数。 6. flfloat 浮点数用于表示 32 位浮点数。 7. double Double 用于表示 64 位浮点数。 8. char Char 用于表示 16 位字符。 9. String 字符串用于表示字符串。 S.No. 布尔运算符 描述 1 AND 与 2 OR 或 3 NOT 非 2.5 CREATE 创建多个标签 2.6 WHERE 子句 where 中的比较运算符 和 之前 mysql 的相同 如 = != <> > < 等 2.7 DELETE 子句 和 REMOVE 子句 DELETE 子句 删除节点。 CREATE (<node-name>:<label-name1>:<label-name2>.....:<label-namen>) 如 : CREATE (person:Person:Beauty:Picture {cid:20,name:" 小美女 "}) 简单的 WHERE 子句 WHERE <condition> 复杂的 WHERE 子句 WHERE <condition> < boolean -operator> <condition> MATCH (person:Person) WHERE person.name = ' 范闲 ' OR person.name = ' 靖王世子 ' RETURN person 删除节点及相关节点和关系。 REMOVE 子句 删除节点或关系的标签 删除节点或关系的属性 2.8 SET 子句 向现有节点或关系添加新属性 更新属性值 2.9 ORDER BY 子句 “ORDER BY” 子句,对 MATCH 查询返回的结果进行排序。 我们可以按升序或降序对行进行排序。 默认情况下,它按升序对行进行排序。 如果我们要按降序对它们进行排序,我们需要使用 DESC 子句。 2.10 SKIP LIMIT Neo4j CQL 已提供 “SKIP” 子句来过滤或限制查询返回的行数。 它修整了 CQL 查询结果集顶部的结果。 Neo4j CQL 已提供 “LIMIT” 子句来过滤或限制查询返回的行数。 它修剪 CQL 查询结果集底部的结果。 2.11 DISTINCT 排重 这个函数的用法就像 SQL 中的 distinct 关键字,返回的是所有不同值。 match p = (:Person {name:" 林婉儿 "})-[r:Couple]-(:Person) delete r MATCH (person:Person {name:" 小美女 "}) REMOVE person.cid MATCH (person:Person {cid:1}) SET person.money = 3456 ,person.age= 25 MATCH (person:Person) RETURN person.name,person.money ORDER BY person.money DESC MATCH (person:Person) RETURN ID(person),person.name,person.money ORDER BY person.money DESC skip 4 limit 2 S.No. 功能 描述 1 UPPER 它用于将所有字母更改为大写字母。 2 LOWER 它用于将所有字母改为小写字母。 3 SUBSTRING 它用于获取给定 String 的子字符串。 4 REPLACE 它用于替换一个字符串的子字符串。 S.No. 聚集功能 描述 1 COUNT 它返回由 MATCH 命令返回的行数。 2 MAX 它从 MATCH 命令返回的一组行返回最大值。 3 MIN 它返回由 MATCH 命令返回的一组行的最小值。 4 SUM 它返回由 MATCH 命令返回的所有行的求和值。 5 AVG 它返回由 MATCH 命令返回的所有行的平均值。 第三部分 Neo4j CQL 高级 3.1 CQL 函数 3.1.1 字符串函数 3.1.2 聚合函数 3.1.3 关系函数 MATCH (p:Person) RETURN Distinct(p.character) MATCH (p:Person) RETURN ID(p),LOWER(p.character) match(p:Person) return p.character,lower(p.character),p.name,substring(p.name,2),replace(p.name," 子 ","z i") MATCH (p:Person) RETURN MAX(p.money),SUM(p.money) S.No. 功能 描述 1 STARTNODE 它用于知道关系的开始节点。 2 ENDNODE 它用于知道关系的结束节点。 3 ID 它用于知道关系的 ID 。 4 TYPE 它用于知道字符串表示中的一个关系的 TYPE 。 3.1.4 shortestPath 函数返回最短的 path 3.2 CQL 多深度关系节点 1. 使用 with 关键字 查询三层级关系节点如下: with 可以将前面查询结果作为后面查询条件 match (na:Person)-[re]->(nb:Person) where na.name=" 范闲 " WITH na,re,nb match (nb:Person)- [re2]->(nc:Person) return na,re,nb,re2,nc match (na:Person)-[re]->(nb:Person) where na.name=" 林婉儿 " WITH na,re,nb match (nb:Person)- [re2]->(nc:Person) return na,re,nb,re2,nc match (na:Person)-[re]-(nb:Person) where na.name=" 林婉儿 " WITH na,re,nb match (nb:Person)- [re2]->(nc:Person) return na,re,nb,re2,nc match (na:Person)-[re]-(nb:Person) where na.name=" 林婉儿 " WITH na,re,nb match (nb:Person)- [re2:Friends]->(nc:Person) return na,re,nb,re2,nc 2. 直接拼接关系节点查询 match (na:Person{name:" 范闲 "})-[re]->(nb:Person)-[re2]->(nc:Person) return na,re,nb,re2,nc 为了方便,可以将查询结果赋给变量,然后返回 match data=(na:Person{name:" 范闲 "})-[re]->(nb:Person)-[re2]->(nc:Person) return data 3. 使用深度运算符 当实现多深度关系节点查询时,显然使用以上方式比较繁琐。 可变数量的关系 -> 节点可以使用 -[:TYPE*minHops..maxHops]- 。 查询: match data=(na:Person{name:" 范闲 "})-[*1..2]-(nb:Person) return data match p = (:Person {name:" 林婉儿 "})-[r:Couple]-(:Person) RETURN STARTNODE(r) MATCH p=shortestPath( (node1)-[*]-(node2) ) RETURN length(p), nodes(p) MATCH p=shortestPath((person:Person {name:" 王启年 "})-[*]-(person2:Person {name:" 九品射手燕小乙 "}) ) RETURN length(p), nodes(p) 3.3 事务 为了保持数据的完整性和保证良好的事务行为, Neo4j 也支持 ACID 特性 。 注意 : 3.4 索引 3.4.1 简介 Neo4j CQL 支持节点或关系属性上的索引,以提高应用程序的性能。 可以为具有相同标签名称的属性上创建索引。 可以在 MATCH 或 WHERE 等运算符上使用这些索引列来改进 CQL 的执行。 3.4.2 创建单一索引 CREATE INDEX ON :Label(property) 例如 : CREATE INDEX ON :Person(name) 3.4.3 创建复合索引 CREATE INDEX ON :Person(age, gender) 3.4.4 全文模式索引 之前的常规模式索引只能对字符串进行精确匹配或者前后缀索引 (startswith,endswith,contains) ,全文 索引将标记化索引字符串值,因此它可以匹配字符串中任何位置的术语。索引字符串如何被标记化并分 解为术语,取决于配置全文模式索引的分析器。索引是通过属性来创建,便于快速查找节点或者关系。 创建和配置全文模式索引 使用 db.index.fulltext.createNodeIndex 和 db.index.fulltext.createRelationshipIndex 创建全文模式索 引。在创建索引时,每个索引必须为每个索引指定一个唯一的名称,用于在查询或删除索引时引用相关 的特定索引。然后,全文模式索引分别应用于标签列表或关系类型列表,分别用于节点和关系索引,然 后应用于属性名称列表。 (1) 所有对 Neo4j 数据库的数据修改操作都必须封装在事务里。 (2) 默认的 isolation level 是 READ_COMMITTED 。 (3) 死锁保护已经内置到核心事务管理 。 (Neo4j 会在死锁发生之前检测死锁并抛出异常。在异常抛出之 前,事务会被标志为回滚。当事务结束时,事务会释放它所持有的锁,则该事务的锁所引起的死锁也就是解 除,其他事务就可以继续执行。当用户需要时,抛出异常的事务可以尝试重新执行 ) (4) 除特别说明 ,Neo4j 的 API 的操作都是线程安全的, Neo4j 数据库的操作也就没有必要使用外部的同步方 法。 call db.index.fulltext.createNodeIndex( " 索引名 " ,[Label,Label],[ 属性 , 属性 ]) call db.index.fulltext.createNodeIndex( "nameAndDescription" ,[ "Person" ],[ "name" , "description" ]) 3.4.5 查看和删除索引 call db.indexes 或者 :schema DROP INDEX ON :Person(name) DROP INDEX ON :Person(age, gender) call db.index.fulltext.drop("nameAndDescription") 3.5 约束 3.5.1 唯一性约束 作用 避免重复记录。 强制执行数据完整性规则 创建唯一性约束 CREATE CONSTRAINT ON ( 变量 :<label_name>) ASSERT 变量 .<property_name> IS UNIQUE 具体实例: CREATE CONSTRAINT ON (person:Person) ASSERT person.name IS UNIQUE 删除唯一性约束 DROP CONSTRAINT ON (cc:Person) ASSERT cc.name IS UNIQUE 3.5.2 属性存在约束 ( 企业版中可用 ) 3.5.3 查看约束 第四部分 Neo4j Admin 管理员操作 4.1 Neo4j - 数据库备份和恢复 在对 Neo4j 数据进行备份、还原、迁移的操作时,首先要关闭 neo4j call db. index .fulltext.queryNodes("nameAndDescription", " 范闲 ") YIELD node, score RETURN node.name, node.description, score CREATE CONSTRAINT ON (p:Person) ASSERT exists(p.name) call db.constraints :schema 数据备份到文件 还原、迁移之前 ,关闭 neo4j 服务。操作同上 重启服务 注意,运行数据备份可能会警告 WARNING: Max 1024 open fifiles allowed, minimum of 40000 recommended. See the Neo4j manual 1. 编辑这个文件 vi /etc/security/limits.conf 在文件最后加入下面这段 修改最大打开文件限制 2. 重启服务器 再次执行上面的步骤 警告就没有了 4.2 调优思路 1. 增加服务器内存 和 调整 neo4j 配置文件 2.neo4j 刚启动数据是冷的需要预热 3. 查看执行计划进行索引优化 ./bin/neo4j stop ./bin/neo4j-admin dump --database=graph.db --to=/root/qyn.dump ./bin/neo4j-admin load --from=/root/qyn.dump --database=graph.db --force ./bin/neo4j start * soft nofile 65535 * hard nofile 65535 # java heap 初始值 dbms.memory.heap.initial_size=1g # java heap 最大值,一般不要超过可用物理内存的 80 % dbms.memory.heap.max_size=16g # pagecache 大小,官方建议设为: ( 总内存 -dbms.memory.heap.max_size)/2 , dbms.memory.pagecache.size=2g MATCH (n) OPTIONAL MATCH (n)-[r]->() RETURN count(n.name) + count(r); Cypher 查询计划程序将每个查询转换为执行计划。 执行计划告诉 Neo4j 在执行查询时要执行哪些操 作。 对执行计划的生成, Neo4j 使用的都是基于成本的优化器( Cost Based Optimizer , CBO ),用于制订 精确的执行过程。可以采用如下两种不同的方式了解其内部的工作机制: EXPLAIN :是解释机制,加入该关键字的 Cypher 语句可以预览执行的过程但并不实际执行,所以也不 会产生任何结果。 PROFILE :则是画像机制,查询中使用该关键字,不仅能够看到执行计划的详细内容,也可以看到查询 的执行结果。 使用索引和不使用索引对比 MATCH (p { name : ' 范闲 ' }) RETURN p 在之前加上 profifile 来进行查询,可以查看查询计划 第五部分 Neo4j 程序访问 5.1 Neo4j 数据库访问 5.1.1 Neo4j 访问的两种方式 嵌入式数据库 服务器模式 ( 通过 REST 的访问 ) 它是由应用程序的性质( neo4j 是独立服务器 还是和程序在一起 ), 性能,监控和数据安全性来决定架构 选择。 5.1.2 An embedded database (嵌入式数据库) 嵌入式 Neo4j 数据库是性能的最佳选择。 通过指定数据存储的路径以编程方式访问嵌入式数据库。 我们选择嵌入式数据库出于以下原因: 使用 Java 作为我们项目的编程语言时 应用程序是独立的 程序追求很高的性能 5.1.3 Neo4j Server (服务器模式) Neo4j Server 是相互操作性,安全性和监控的最佳选择。 实际上, REST 接口允许所有现代平台和编程 语言与它进行互操作。 此外,作为独立应用程序,它比嵌入式配置更安全(客户端中的潜在故障不会影 响服务器),并且更易于监控。 如果我们选择使用这种模式,我们的应用程序将充当 Neo4j 服务器的客 户端。要连接到 Neo4j 服务器,可以使用任何编程语言的 REST 访问数据库。 5.2 Java 客户端操作 Neo4j 1. 嵌入式模式 关注指标: estimated rows : 需要被扫描行数的预估值 dbhits : 实际运行结果的命中绩效 两个值都是越小越好 <dependency> <groupId> org.neo4j </groupId> <artifactId> neo4j </artifactId> <version> 3.5.5 </version> </dependency> package com . lagou ; import org . neo4j . graphdb . * ; import org . neo4j . graphdb . factory . GraphDatabaseFactory ; import java . io . File ; import java . util . HashMap ; import java . util . Map ; public class EmbeddedNeo4jAdd { private static final File databaseDirectory = new File ( "target/graph.db" ); public static void main ( String [] args ) { GraphDatabaseService graphDb = new GraphDatabaseFactory (). newEmbeddedDatabase ( databaseDirectory ); System . out . println ( "Database Load!" ); Transaction tx = graphDb . beginTx (); Node n1 = graphDb . createNode (); n1 . setProperty ( "name" , " 张三 " ); n1 . setProperty ( "character" , "A" ); n1 . setProperty ( "gender" , 1 ); n1 . setProperty ( "money" , 1101 ); n1 . addLabel ( new Label () { @Override public String name () { return "Person" ; } }); String cql = "CREATE (p:Person{name:' 李 四 ',character:'B',gender:1,money:21000})" ; graphDb . execute ( cql ); tx . success (); tx . close (); System . out . println ( "Database Shutdown!" ); graphDb . shutdown (); } } package com . lagou ; import org . neo4j . graphdb . * ; import org . neo4j . graphdb . factory . GraphDatabaseFactory ; import java . io . File ; import java . util . HashMap ; import java . util . Map ; public class EmbeddedNeo4jQueryAll { 2. 服务器模式 private static final File databaseDirectory = new File ( "target/graph.db" ); public static void main ( String [] args ) { GraphDatabaseService graphDb = new GraphDatabaseFactory (). newEmbeddedDatabase ( databaseDirectory ); System . out . println ( "Database Load!" ); String cql = "MATCH (a:Person) where a.money < $money return a" ; Map < String , Object > paramerters = new HashMap < String , Object > (); paramerters . put ( "money" , 25000 ); Transaction tx = graphDb . beginTx (); Result result = graphDb . execute ( cql , paramerters ); while ( result . hasNext ()) { Map < String , Object > row = result . next (); for ( String key : result . columns ()) { Node nd = ( Node ) row . get ( key ); System . out . printf ( "%s = %s:%s%n" , key , nd . getProperty ( "name" ), nd . getProperty ( "money" )); } } tx . success (); tx . close (); System . out . println ( "Database Shutdown!" ); graphDb . shutdown (); } } <dependency> <groupId> org.neo4j </groupId> <artifactId> neo4j-ogm-bolt-driver </artifactId> <version> 3.2.10 </version> </dependency> package com . lagou ; import org . neo4j . driver . * ; import static org . neo4j . driver . Values . parameters ; public class Neo4jServerMain { public static void main ( String [] args ) { Driver driver = GraphDatabase . driver ( "bolt://127.0.0.1:7687" , AuthTokens . basic ( "neo4j" , "123456" ) ); Session session = driver . session (); String cql = "MATCH (a:Person) WHERE a.money > $money " + "RETURN a.name AS name, a.money AS money order by a.money " ; Result result = session . run ( cql , 5.3 SpringBoot 整合 Neo4j parameters ( "money" , 1000 ) ); while ( result . hasNext () ) { Record record = result . next (); System . out . println ( record . get ( "name" ). asString () + " " + record . get ( "money" ). asDouble () ); } session . close (); driver . close (); } } package com . lagou ; import org . neo4j . driver . * ; import static org . neo4j . driver . Values . parameters ; public class Neo4jServerMain2 { public static void main ( String [] args ) { Driver driver = GraphDatabase . driver ( "bolt://127.0.0.1:7687" , AuthTokens . basic ( "neo4j" , "123456" ) ); Session session = driver . session (); String cql = "MATCH p=shortestPath((person:Person {name:$startName})- [*]-(person2:Person {name:$endName} )) RETURN p" ; Result result = session . run ( cql , parameters ( "startName" , " 王启 年 " , "endName" , " 九品射手燕小乙 " ) ); while ( result . hasNext () ) { Record record = result . next (); System . out . println ( record ); } session . close (); driver . close (); } } 1. 导入 jar <?xml version="1.0" encoding="UTF-8"?> <project xmlns = "http://maven.apache.org/POM/4.0.0" xmlns:xsi = "http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation = "http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd" > <modelVersion> 4.0.0 </modelVersion> <groupId> com.lagou </groupId> <artifactId> neo4j_springboot_demo </artifactId> <version> 1.0-SNAPSHOT </version> 2. 建立实体类 <parent> <groupId> org.springframework.boot </groupId> <artifactId> spring-boot-starter-parent </artifactId> <version> 2.0.5.RELEASE </version> </parent> <properties> <project.build.sourceEncoding> UTF-8 </project.build.sourceEncoding> <project.reporting.outputEncoding> UTF- 8 </project.reporting.outputEncoding> <java.version> 1.11 </java.version> </properties> <dependencies> <dependency> <groupId> org.springframework.boot </groupId> <artifactId> spring-boot-starter-data-neo4j </artifactId> </dependency> <dependency> <groupId> org.neo4j </groupId> <artifactId> neo4j-ogm-bolt-driver </artifactId> </dependency> </dependencies> <build> <plugins> <plugin> <groupId> org.springframework.boot </groupId> <artifactId> spring-boot-maven-plugin </artifactId> </plugin> </plugins> </build> </project> package com . lagou . bean ; import org . neo4j . ogm . annotation . GeneratedValue ; import org . neo4j . ogm . annotation . Id ; import org . neo4j . ogm . annotation . NodeEntity ; import org . neo4j . ogm . annotation . Property ; @NodeEntity public class Person { @Id @GeneratedValue private Long id ; @Property ( "cid" ) private int pid ; @Property private String name ; private String character ; private double money ; private int gender ; private int age ; private String description ; @Relationship ( type = "Friends" , direction = Relationship . INCOMING ) private Set < Person > relationPersons ; @Override public String toString () { return "Person{" + "id=" + id + ", pid=" + pid + ", name='" + name + '\'' + ", character='" + character + '\'' + ", money=" + money + ", gender=" + gender + ", age=" + age + ", description='" + description + '\'' + ", relationPersons=" + relationPersons + '}' ; } public Person () { } public Person ( Long id , String name , String character , double money , int gender , int age , String description ) { this . id = id ; this . name = name ; this . character = character ; this . money = money ; this . gender = gender ; this . age = age ; this . description = description ; } public Set < Person > getRelationPersons () { return relationPersons ; } public void setRelationPersons ( Set < Person > relationPersons ) { this . relationPersons = relationPersons ; } public Long getId () { return id ; } public void setId ( Long id ) { this . id = id ; } public String getName () { return name ; } public void setName ( String name ) { this . name = name ; 3. 数据持久化类 } public String getCharacter () { return character ; } public void setCharacter ( String character ) { this . character = character ; } public double getMoney () { return money ; } public void setMoney ( double money ) { this . money = money ; } public int isGender () { return gender ; } public void setGender ( int gender ) { this . gender = gender ; } public int getAge () { return age ; } public void setAge ( int age ) { this . age = age ; } public String getDescription () { return description ; } public void setDescription ( String description ) { this . description = description ; } } package com . lagou . repository ; import com . lagou . bean . Person ; import org . springframework . data . neo4j . annotation . Query ; import org . springframework . data . neo4j . repository . Neo4jRepository ; import org . springframework . stereotype . Repository ; 4. 配置文件 application.yml 5. 编写服务类 import java . util . * ; @Repository public interface PersonRepository extends Neo4jRepository < Person , Long > { @Query ( "match(p:Person) where p.money > {0} return p" ) List < Person > personList ( double money ); @Query ( "MATCH p=shortestPath((person:Person {name:{0}})-[*1..4]- (person2:Person {name:{1}}) ) RETURN p" ) List < Person > shortestPath ( String startName , String endName ); } 或者使用 @Query("match(p:Person) where p.money>{money} return p") List<Person> personList(@Param("money") double money); /** 指定开始的名字 和 结束的名字 查询最短路径 限定深度为 4 以层包含 4*/ @Query("match p=shortestPath((person:Person{name:{startName}})-[*1..4]- (person2:Person {name:{endName}})) return p") List<Person> shortestPath(@Param("startName") String startName,@Param("endName") String endName); spring : data : neo4j : username : neo4j password : 123456 uri : bolt : //192.168.211.133 : 7687 #uri: http://192.168.211.133:7474 #uri: file:///target/graph.db package com . lagou . service ; import com . lagou . bean . Person ; import com . lagou . repository . PersonRepository ; import org . springframework . beans . factory . annotation . Autowired ; import org . springframework . stereotype . Service ; import java . util . * ; @Service public class Neo4jPersonService { @Autowired private PersonRepository personRepository ; public List < Person > personList (){ return personRepository . personList (); } public Person save ( Person person ){ return personRepository . save ( person ); } 6. 编写测试类 public List < Person > shortestPath ( String startName , String endName ){ return personRepository . shortestPath ( startName , endName ); } package com . lagou ; import com . lagou . bean . Person ; import com . lagou . service . Neo4jPersonService ; import org . springframework . boot . SpringApplication ; import org . springframework . boot . autoconfigure . SpringBootApplication ; import org . springframework . context . ApplicationContext ; import java . util . List ; @SpringBootApplication public class TestNeo4jBootApp { public static void main ( String [] args ) { ApplicationContext app = SpringApplication . run ( TestNeo4jBootApp . class , args ); Neo4jPersonService personService = app . getBean ( Neo4jPersonService . class ); System . out . println ( personService ); List < Person > datas = personService . personListAll (); System . out . println ( datas ); System . out . println ( personService . shortestPath ( " 王启年 " , " 九品射手燕小 乙 " )); } }
上一篇:ORACLE 11G sqlplus连接报错ORA-09925: Unable to create audit trail file Linux-x86_64 Error: 28: No space


下一篇:Wordpress 删除 Storefront 主题的购物车