SSH架构中paramsPrepareParamsStack拦截器的PO id注入问题

Struts2中的paramsPrepareParamsStack拦截器中,第一个params拦截器自动将request的parameter值注入到Action的instance变量以及其关联对象中,然后我们可以在Action中的prepare方法中根据id从数据库将PO实体装载进来,后一个params拦截器中再将新值赋给这些PO对象,这样就保证了数据库原有的一些属性值不会丢失。

这一功能很好很强大,但在与OpenSessionInView结合时,会有一个小陷阱需要防范。

举一个典型的应用场景:

一个Group对象,有id,name等属性
一个User对象,与Group是多对一的关系,User有id,name和group等属性

在修改后保存User时,form表单的input如下:

<input type="text" name="user.id" .../>
<input type="text" name="user.name" .../>
<select name="user.group.id">
<option value=1>group1</option>
<option value=2>group2</option>
</select>


Action的prepare方法如下:
	public void prepare() throws Exception {
if (user != null && user.getId() != null) {
user = userManager.get(user.getId());
}
}


某一User的group原来为group1,修改为group2后,点保存,Action执行了updateUser()方法,Hibernate会报错:

org.springframework.orm.hibernate3.HibernateSystemException: identifier of an instance of com.xxx.model.Group was altered from 1 to 2; nested exception is org.hibernate.HibernateException: identifier of an instance of com.xxx.model.Group was altered from 1 to 2


原来,prepare方法调用后,user对象被重置为hibernate生成的po,user.getGroup()的group对象处于persistent状态,它的id来自于数据库,为1。第二个param拦截器执行了user.getGroup.setId(2)操作,所以抛出了上述异常。

解决办法是,在prepareUpdateUser()方法中执行:
public void prepareUpdateUser() throws Exception {
if (user != null && user.getId() != null) {
long newGroupId = user.getGroup().getId();
user = userManager.get(user.getId());
if (newGroupId != user.getGroup().getId()) {
Group newGroup = new Group();
newGroup.setId(newGroupId);
user.setGroup(newGroup);
}
}
}

这样在group修改后手动将与user关联的处于persistent状态的group断开联系。

更简单的方式是直接
public void prepareUpdateUser() throws Exception {
if (client != null && client.getId() != null) {
client = clientManager.get(client.getId());
client.setGroup(null);
}
}

因为第二个param拦截器还会正确地把不管是新的还是老的group id注入进来。

注:该方式只适用于user不对group进行级联更新的情况
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值