假设我有两个不同的线程T1和T2,它们同时访问同一个数据库并从同一个表中获取数据。如何避免两个不同的线程从数据库读取相同的行(Hibernate和Oracle 10g)
现在在线程启动时,我需要从表中获取数据并将行存储到集合中,然后我将使用它们在别处执行某些工作。我不希望这两个线程能够处理相同的数据,因为它会导致重复(和长时间)的工作。更具体地说,这是一个企业应用程序,需要在启动时加载一些记录并将其存储在集合中以完成一些额外的工作。问题是在集群环境中,这可能会导致两个不同的实例加载相同的数据,因此工作可能会重复。所以我希望行只能由一个实例加载一次。
我该如何避免这种情况?
我目前使用Hibernate和Oracle 10g。这些是我的解决方案到现在:
以编程方式锁定行。第一个读取它会将一些“锁定”列设置为true,但如果第一个线程死亡而没有将行设置为“已处理”,很可能会发生死锁。
使用悲观锁定。我尝试使用LockMode.UPGRADE,但似乎没有帮助,因为我仍然能够同时从两个线程读取数据。
public List<MyObject> getAllNtfFromDb() { Session session = HibernateUtil.getOraclesessionfactory().openSession(); Query q = session.createQuery( "from MyObject n where n.state = 'NEW'"); List<MyObject> list = (List<MyObject>) q.list(); for (int i=0; i<list.size(); i++) session.lock(list.get(i), LockMode.UPGRADE); return list; }
任何其他提示吗?我究竟做错了什么?
谢谢。
也许你可以使用select for update语句? – soilworker 2015-01-15 17:09:29