Neo4j Merge和Atomic Transaction

时间:2015-12-16 00:49:52

标签: neo4j atomic graph-databases neo4jclient

我在10个并行线程中从客户端应用程序对我的Neo4j服务器运行以下MERGE查询,newFooid参数在所有10次运行中都是相同的:

MERGE (foo:Foo { id: {id} })
ON MATCH
SET foo = {newFoo}

运行此操作后,我运行以下查询以期待1,但我改为10

match (f:Foo)
return count(f)

我认为MERGE在原子事务中运行但显然不是。我在这里做错了吗?

更新

以下是我用来重现此问题的代码:

public static async Task RunInParallel()
{
    var client = new GraphClient(new Uri("http://localhost:7474/db/data"), "neo4j", "1234567890")
    {
        JsonContractResolver = new CamelCasePropertyNamesContractResolver()
    };

    client.Connect();

    var foo = new Foo
    {
        Id = "1",
        Name = "Foo",
        Slug = "foo-bar-foo"
    };

    List<Task> tasks = new List<Task>();
    for (int i = 0; i < 10; i++)
    {
        var task = client.Cypher
            .Merge("(foo:Foo { id: {id} })")
            .OnMatch()
            .Set("foo = {newFoo}")
            .WithParams(new
            {
                Id = foo.Id,
                NewFoo = foo
            })
            .ExecuteWithoutResultsAsync();

        tasks.Add(task);
    }

    await Task.WhenAll(tasks.ToArray());
}

1 个答案:

答案 0 :(得分:5)

MERGE(单独)并不保证唯一性。使用MERGE(在唯一属性上)时,您应始终为指定的属性创建uniqueness constraint

CREATE CONSTRAINT ON (f:Foo) ASSERT f.id IS UNIQUE

这将确保您不会创建任何重复项。

修改

没有唯一性约束的

MERGE不是线程安全的。添加唯一性约束可确保在写入之前保持索引锁定,从而使操作线程安全。