【问题标题】:synchronized method for accessing database with spring and hibernatespring和hibernate访问数据库的同步方法
【发布时间】:2019-11-15 18:53:39
【问题描述】:

我有一个表,该表维护一个序列号,该序列号用作多个表的标识符(多个发票表,所有表都使用单个序列)。 每当我想在发票表中插入新记录时,我都会从表中读取当前序列号并用 +1 更新它。 问题是当有多个新发票号请求时,序列号会返回重复的数字。我尝试了同步块,但当多个请求同时命中时,它仍然返回重复值。

这里是获取序列号的方法

synchronized public int getSequence(){
  Sequence sequence = getCurrentSession().get(Sequence.class,1); //here 1 is the id of the row
  int number = sequence.getSequenceNumber();
  sequence.setSequenceNumber(number+1);
  getCurrentSession().saveOrUpdate(sequence);
  return number;
}

我有什么遗漏吗?

【问题讨论】:

  • 您应该在单独的事务中获得序列中的下一个数字。但是为了节省更多,请查看@GeneratedValue(strategy = GenerationType.SEQUENCE)
  • synchronized 在您的服务器应用程序将运行多个副本时毫无用处。使用正确的事务(Spring 让这变得非常容易)。
  • 永远不要手动更新序列。使用 db sewuence、autogenerate 或与 jpa 一起考虑 uuid 作为解决方案(也许是最好的)
  • 我的情况不同。序列号在所有三个表上都维护。它对所有3个表都是唯一的。!
  • 然后事务T2会因为冲突而回滚,不得不重启。

标签: java spring hibernate synchronized


【解决方案1】:

首先,我不建议您使用序列的表实现。 Explanation why

但是如果你必须 - hibernate 知道如何管理它。 Take a look

还有一件事。我强烈建议您在数据库端实现同步。想象一下,您有 2 个应用程序实例连接到同一个数据库实例并同时工作。

【讨论】:

    【解决方案2】:

    使用事务也不适合我。我尝试了 mysql 中的所有隔离,但没有任何帮助。我用下面的解决方案解决了。

    synchronized public int getSequence(){
       Sequence sequence = getCurrentSession().get(Sequence.class,1); //here 1 is the id of the row
       int prevNumber = sequence.getSequenceNumber();
       Query<Sequence> query = getCurrentSession().createQuery("UPDATE Sequence SET sequenceNumber = :number WHERE sequenceNumber = :prevNumber",Sequence.class);
       query.setParameter("number",prevNumber+1);
       query.setParameter("prevNumber",prevNumber);
       int affectedRows = query.executeUpdate();
       if(accectedRows > 0)
          return number;
       else
         throw new Exception();
    }
    

    所以每当发生冲突时都会抛出异常。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2012-03-10
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多