在具有双向关系的User
之间创建关系的正确方法是什么?
说我有:
class User
include Neo4j::ActiveNode
property :name, type: String
property :created_at, type: DateTime
property :updated_at, type: DateTime
has_many :both, :friends, model_class: 'User', type: 'connection' unique: true
has_many :in, :followers, model_class: 'User', type: 'connection', unique: true
has_many :out, :following, model_class: 'User', type: 'connection', unique: true
end
然后,
users = [
User.create(name: 'Foo'),
User.create(name: 'Bar'),
User.create(name: 'Baz'),
]
这是否适合这样做?这看起来非常低效:
users.each do |user|
user.friends << users.reject { |u| u == user }
end
答案 0 :(得分:2)
如果您想确保user.friends
返回其他两个用户,我实际上是这样做的:
users.each do |user|
other_users = users.reject { |u| u == user || user.friends.include?(u) }
user.friends << other_users
end
现在 非常低效!它创建一个新数组,并为每个用户对数据库执行额外查找。 (FWIW,include?
查询非常快。您也可以删除reject
并再次遍历users
以加快速度。)但仍然需要完成这是因为否则,您将创建重复的关系。
在您的关联中设置unique: true
只会将您保存在一个方向。针对has_many :both
关联的查询使用与方向无关的Cypher MATCH
,但CREATE
操作始终需要方向,因此它们从左侧的节点转到右侧的节点。 user1.friends << user2
和user2.friends << user1
将在两个节点之间创建两个关系,每个方向一个。
修改强>
正如您在your own comment中针对Neo4j.rb上的问题所指出的那样,使用user.friends = [other_friends]
将始终清除所有现有关系并防止重复。这是解决问题的一种非常酷的方式。