关于hibernate 事务提交的问题

近期做项目,遇到了一个问题,用hibernate做数据持久化,可是有的业务逻辑要求是这样的,三张表是独立的。
如何将三张表的增删改放到一个事务里面?
比如:我先保存一个新的人员信息。然后再查询出保存完之后的人员信息同时要update另外一张表里面。
问题就是:session.save(obj) 之后再查询的话。这个对象还是游离状态。会查询不到。
所以就不能后面的update操作

[quote]session.save() 一个对象的时候,没有commit() 事务 可以查到这个对象生成的ID 吗? [/quote]

:可以, see:
[code="java"]
Session s = openSession();
Transaction tx = s.beginTransaction();
int testLength = 3;
Long lastId = null;
for (int i = 0; i < testLength ; i++) {
Car car = new Car();
car.setColor( "color " + i );
s.save( car );
lastId = car.getId();
}[/code]

[quote]你好!persist 用这个方法 先保存新对象,然后再 用getter 能得到新生成的这个对象ID 么?[/quote]

可以, see:

[code="java"] Session s = openSession();
Transaction tx = s.beginTransaction();
Radio radio = new Radio();
radio.setFrequency("32 MHz");
s.persist(radio);
assertEquals( new Integer(1), radio.getId() );
radio = new Radio();
radio.setFrequency("32 MHz");
s.persist(radio);
assertEquals( new Integer(2), radio.getId() );
tx.commit();
s.close();[/code]

……
session.save(obj)之后根本不用查询,传入的obj本身就是托管状态了,可以直接用。

在同一个session里怎么会查询不到呢

[code="java"]
trans = session.beginTransaction(); //开始事务
session.save(t);
Query query = session.createQuery(hql);
trans.commit(); //提交事务
[/code]

首先你的那个对象是个新的对象是临时对象,然后你要吧这个对象update到另外一个表中,那个表中应该就是有这个数据的他是持久状态的。反正我是晕了。
你要是说保存这个对象的同时要拿到这个对象所生成的Id,然后在插入到另外一张表的话,你可以直接使用的保存的那个对象就可以getter他生成的Id了。建议在事务中使用persist来实现持久化别用save。

[quote]
你好,session.save() 一个对象的时候,没有commit() 事务 可以查到这个对象生成的ID 吗?
可能我上面的问题 没有描述清楚。谢谢
[/quote]
我记得是可以的,不过这种其实不用再问了,自己测试一下不就行了……

[quote]session.save() 一个对象的时候,没有commit() 事务 可以查到这个对象生成的ID 吗? [/quote]
可以的。

[code="java"]
trans = session.beginTransaction(); //开始事务

session.save(a);

b.setParentId(a.getId);
session.save(b);
c.setParentId(b.getId);
session.save(c);
trans.commit();
[/code]

首先,先说一下主键ID的生成跟事务的关系,就是主键ID是不在事务里面的(除非你自己设置其值),如果添加一个对像,但没有提交事务,这个对像的ID是会有值的,然后你又回滚了事务,这时你再次查这个主键ID的下一个值(比如MYSQL的AUTO或ORACLE的SEQUENCE)已经发生变化了,也就是说这个主键ID下次生成的值并不会是你回滚前的值了。

因此,你在保存一个未提交的对像后,是可以拿到他的ID给下一次操作使用的,如果在这一系列操作中回滚了事务,那主键ID是以最新值开始(也就是中间跳空了很多,并不连续了)。