Neo4j 3.0.6和neo4j-ogm 2.0.5 - 在数据库上创建重复数据

时间:2016-10-20 12:19:41

标签: neo4j graph-databases spring-data-neo4j-4 neo4j-ogm

我的neo4j数据库有问题。当我尝试初始化数据时,应该只创建一个样本数据,但有时当我尝试初始化数据时,它会创建双样本数据。关于第二次呼叫没有任何痕迹。这是我的Neo4j的配置

@Configuration
@EnableNeo4jRepositories(basePackages = "com.example.neo.repository")
@EnableTransactionManagement
public class Neo4jConfig extends Neo4jConfiguration {
    @Override
    @Bean
    public SessionFactory getSessionFactory() {
        // with domain entity base package(s)
        return new SessionFactory("com.example.neo.model", "BOOT-INF.classes.com.example.neo.model");
    }

    // needed for session in view in web-applications
    @Override
    @Bean
    @Scope(value = "session", proxyMode = ScopedProxyMode.TARGET_CLASS)
    public Session getSession() throws Exception {
        return super.getSession();
    }
}

这就是我调用我的功能的方式

@RequestMapping(value = "/initCurrency")
public ModelAndView initCurrency() {
    initializationService.initCurrency();

    ModelAndView model = new ModelAndView("redirect:/");
    return model;
}

这是initializationService函数

private String[][] currencyList = {
        { "USD", "7.5" },
        { "DKK", "1" },
        { "AFN", "1"},{ "EUR", "1"},{ "ALL", "1"},{ "DZD", "1"},{ "USD", "1"},{ "AOA", "1"},{ "XCD", "1"},
        { "ARS", "1"},{ "AMD", "1"},{ "AWG", "1"},{ "SHP", "1"},{ "AUD", "1"},{ "AZN", "1"},{ "BSD", "1"},
        { "BHD", "1"},{ "BDT", "1"},{ "BBD", "1"}
}
@Override
public void initCurrency() {
    for (String[] currency : currencyList) {
        Currency existCurrency = currencyService.findByName(currency[0]);

        if (existCurrency == null) {
            existCurrency = new Currency(currency[0], Double.valueOf(currency[1]));
            currencyService.save(existCurrency);
        }
    }
}

1 个答案:

答案 0 :(得分:2)

避免重复的唯一可靠方法是在属性上有一个实际的unicity constraint

CREATE CONSTRAINT ON (n:Currency) ASSERT n.name IS UNIQUE;

在SDN 4中没有办法。[0-2](尚未)从模型中创建这样的约束(在SDN 3.x中有@Indexed@Indexed(unique = true)注释),所以你必须独立运行查询(例如使用Liquigraph!)。

在并发环境中进行查找以保护创建是不够的(顺序调用是正常的),因为读取和写入之间没有锁定,这可能导致以下情况与交错执行:

  • 线程A检查是否存在,找不到节点
  • 线程B检查存在,找不到节点
  • 主题A创建名为“USB”的货币节点
  • 主题B创建另一个名称相同的货币节点

由于您最终会出现重复项,因此发生了2次并发调用 。负载均衡器具有非常短的超时并配置为重试?激活HTTP日志,在Spring控制器或服务中添加一些日志记录,使用tcpdump捕获流量等。一旦unicity约束处于活动状态,将更容易隔离第二个调用,因为您将获得异常。