springboot+neo4j 简单的社交网络+爱好权重+好友推荐

spring-boot-neo4j

spring-boot集成neo4j图形数据库,并实现简单的节点创建和查询

这个项目是大二时数据库的课程项目 现在稍微写点readme 发布一下

github地址 https://github.com/YULuoOo/spring-boot-neo4j

简书博客地址 https://www.jianshu.com/u/35c3b7d55046

关于neo4j

  • Neo4j是一个网络——面向网络的数据库——也就是说,它是一个嵌入式的、基于磁盘的、具备完全的事务特性的Java持久化引擎,但是它将结构化数据存储在网络上而不是表中。网络(从数学角度叫做图)是一个灵活的数据结构,可以应用更加敏捷和快速的开发模式。Neo4j是一个高性能的图引擎,该引擎具有成熟和健壮的数据库的所有特性。程序员工作在一个面向对象的、灵活的网络结构下而不是严格、静态的表中——但是他们可以享受到具备完全的事务特性、企业级的数据库的所有好处。在该模型中,以“节点空间”来表达领域数据——相对于传统的模型表、行和列来说,节点空间是很多节点、关系和属性(键值对)构成的网络。关系是第一级对象,可以由属性来注解,而属性则表明了节点交互的上下文。

  • 本次的实现我们看重的是它在涉及到查询遍历两节点关系上表现出的卓越的轻巧快速,首先,图的遍历相较于两个表的遍历要更加有效率,其次,在试用neo4j的过程中我们可以相比mysql更加容易地修改它的属性,好符合不断考虑修改的数据库设计。还有,它能够容纳数据的量多达几十亿个节点和上百亿关系,能够符合社交网络的庞大用户量和错综复杂的用户和其他节点关系的要求。对于之前仅接触过mysql之类关系数据库的我们来说也是一次新的尝试。

需求分析

功能需求

  • 给用户提供一个与相同爱好者交流联系的平台,系统会通过你的标签,对哪些更加有兴趣,经过好友推荐的算法自动推荐更多与你兴趣相投,志同道合的人。同时,用户可以知道你关注的人对哪些方面感兴趣,以及感兴趣程度,还可以看到他所关注的人,进一步扩展社交圈子,与更多的人交流来往,了解更多的知识,发现更多自己以往不了解的趣事。

实现功能:

  • 显示关注、被关注用户
  • 添加自定义标签,设定权值
  • 通过自定义标签显示爱好重合的同好
  • 通过用户名或者标签搜索用户
  • 通过标签和权值推荐最适合该用户的好友
  • 显示关注、爱好、和被关注最多的用户
  • 显示两个用户的共同关注
  • 显示两个用户之间的最短路径

性能需求

  1. 测试过程中,插入10w节点的情况下对于每个tab的关注、被关注的异步刷新(数据库同时进行查询操作)响应快,没有出现体感的延迟。
  2. neo4j的算法遍历加快查询的速度

数据库设计(非关系型)

这种社交网络的系统,用关系型数据库实现起来的话,会很麻烦,而且即使成功实现了,性能也很差,于是想到用非关系型数据库,其中图数据库最合适,利用neo4j数据库实现社交网络的设计。以用户作为节点(节点的属性描述了用户的个人信息包括用户名、年龄、性别)另外兴趣也作为节点存在,当然了用户节点和兴趣节点之间有所区分,关系存在于用户之间和也存在于用户与兴趣之间,用户可以通过他们分别与兴趣之间的关联度来建立他们之间的关系,用户之间还可以通过他们共同的好友来建立关系。

技术栈

Spring Boot(Intelij)
Spring Boot(Intelij)
Thymeleaf
Avalonjs
Bootstrap、jQuery、ajax

数据库

节点部分:

用户User:{id,name,age,sex,password,address,sign}
标签Tag:{name}

关系部分:

关注Follow:{time}
喜欢Like:{count}

以下是只包括用户节点的图,能够显示出复杂的人际关系
人际关系图

图片 1.png

接下来是整个关系网的全貌


图片 2.png

可以看出,要找到和自己共有一个tag/user的user,只需要通过自己的关系箭头找到对应的tag/user,再以对方为起点遍历就可以了。
以rain为标签的所有用户


图片 3.png

下图为爱好的权值的应用,用来做好友推荐,可以看到LIKE了相同爱好的用户,以及LIKE关系的值,代表喜爱程度。

图片 4.png

以rain为标签的所有用户

Spring boot内实体层、控制层、DAO层。

实体层中,有的是NodeEntity 有的是 relationentity,都要对应到数据库里的实现。
图片 5.png

图片 6.png
在repository类中,写cypher语句。
图片 7.png
图片 8.png

系统的框架逻辑:

从前端采用采用ajax访问controller接口。Controller内部使用Respository的方法进行数据库交互。
图片 9.png

图片 10.png

用户关系和节点显示:

界面内,标签页的每个tab都属于一个独立的mscontroller,在js文件中包含特定ms-controller的内容物,包括其id,自定义属性和自定义函数,用户的follow关系和like关系的界面显示是通过avalonjs的ms-controller内的变量list,类似android里面的listview,这里设定为当按下按钮时调用ms-click里面触发特定函数@函数名,将属性里保存的list中该元素的NodeId取出例如{{user.nodeId}}。随后ms-controller内函数中发出ajax请求发到controller的特定地址,通过controller中Userrepository、FollowRepository、RetrieveRepository等等来进行数据库的信息交换和查询,各自返回list重新存入到对应ms-controller的属性的list中,就做到了一次的异步刷新。Window-onload和每次点击按钮,或者切换标签的时候,都可能进行异步刷新,这样能有效保证不发生反应延迟信息不对的情况。

效果如下,使用ms-for


图片 11.png

标签节点(或者说关系)显示


图片 12.png

实现过程

过程概述

  1. 经由组长(本人)提出的通过nosql neo4j来进行社交网络实现这一想法作为开头,选择spring boot作为框架来进行开发
  2. 配置成功neo4j数据库以后确定项目的大致方向,决定首先实现用户注册存入数据库后登录,先实现用户之间互相关注的功能。
  3. 将基础界面写出,写出实体类,实现数据库内存取基础的关注。
  4. 实现资料的修改以及网页上直接刷新资料的显示
  5. 考虑大批量导入csv文件中的用户数据测试数据库的查询性能,编写随机生成用户名、年龄性别的函数和关系函数,自动循环写入txt文件再转成csv导入。
  6. 实现标签类的显示和增改,以及通过标签类对用户的查询
  7. 考虑更进一步的算法,比如用户之间的最短路径、用户的共同关注,将其显示
  8. 更加完善前后端的交互

遇到的问题及解决

  1. Springboot-neo4j框架问题,花了几周配置环境以及熟悉springboot开发框架。

  2. Neo4j使用cypher查询语言,需要对这个语言进行学习。

  3. 在处理图数据库的数据时,遇到数据类型的问题。比如一个查询要返回既有节点类型的数据,又有关系类型的数据。我们的解决方法是,包装一个类,里面既有node类,又有relation类。

  4. Thymeleaf中导入css文件没有效果的问题,后来发现必须要是非常特定的格式并且在properties里面写了特定路径才能够检测到,解决;

  5. Ajax无法发出请求问题,起初发现is not a function,搜索半天在开发者选项里面的控制台的报错发现未被识别,将$改成jQuery后解决;

  6. Div无法居中、宽度无法自由设定问题,发现是由于导入的css文件有冲突所致,利用内部样式表更加优先这一特点暂时解决;

  7. 无法通过controller return String类来重定向,发现当时是restController并且没有使用返回ModelandView这一方法,修改成Controller或者返回存着地址的mav,解决;

  8. 从session中存的属性通过getAttribute获取后,在session未过期就出现了nullptr获取不到session中user实体的问题,每一次取出后都进行一次存,

  9. 无法通过jQuery获取到th:each或者ms-for里面每一个元素,而无法访问元素的属性nodeId,无法传值也无法增删改的问题,通过直接ms-click设置对应函数即可解决;

  10. FollowRepository中的双向关注实现,发现使用双箭头<-[r]->会导致只要有其中一个关系满足就会被查出来的情况,后来发现能够通过(n)-[r]->(m)-[r]->(n)这个语句解决;

系统中的亮点

路径算法

图片 13.png
```
match p = shortestPath((a:User{name:{name3}})-[*..15]->(b:User{name:{name4}})) 
RETURN p
```

共同好友搜索算法

图片 14.png
```
match (a:User)-[:Follow]->(c)<-[:Follow]-(b:User) 
where a.name={name1} and b.name={name2} 
return c
```

好友推荐算法

图片 15.png
Match(n1:User)-[r1:Like]->(t1:Tag),(n2:User)-[r2:Like]->(t1) 
where n1.name='imyonzmo' with count(n2.name) 
as  num,sqrt(sum((toInt(r1.count)-toInt(r2.count))^2)) as aaa,n1,n2,r1,r2,t1  
return n1,aaa,n2,r1,r2,t1,num order by num,aaa limit     10
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,014评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,796评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,484评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,830评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,946评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,114评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,182评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,927评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,369评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,678评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,832评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,533评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,166评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,885评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,128评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,659评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,738评论 2 351

推荐阅读更多精彩内容