简单地说,发生此异常即是一个游离的对象要被持久化(save)时,其ID既要ORM框架为它生成ID值,而此实体的ID却已然有值。
出现问题原因:
当entity类的id类型为Integer的时候 用apache的BeanUtils.copyProperties() 拷贝后 entity 的id 会被赋成0 因为BeanUtils.copyProperties() 会把null值赋成0 主键带着值 往数据库里save 报错 |
解决方法:
public void addOrg(Organization org, Integer parentid) { if(parentid!=null && parentid>0){ org.setParent((Organization)this.getHibernateTemplate().load(Organization.class, parentid)); } this.getHibernateTemplate().save(org);
//设置机构编号 if(parentid!=null && parentid>0){ org.setSn(org.getParent().getId()+"-"+org.getId()); }else{ org.setSn(org.getId().toString()); } } 将persist(org)改成save(org) |
再用jpa+spring+struts2开发的是时候遇到一个问题(采用了注解的方式,xml配置的道理是一样的),当我在注册用户的时候,注册第一个用户没有问题,但注册第二个用户开始就会抛出一个异常:
javax.persistence.PersistenceException: org.hibernate.PersistentObjectException: detached entity passed to persist: com.dalton.domain.User
org.hibernate.ejb.AbstractEntityManagerImpl.throwPersistenceException
(AbstractEntityManagerImpl.java:614)
org.hibernate.ejb.AbstractEntityManagerImpl.persist(AbstractEntityManagerImpl.java:226)
sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
…………………………
在网上找了半天也没找到答案。
后来通过println找了原因,当我第二次插入用户的时候,还没有实例化到数据库就有了ID值,而且这个ID值是我第一次插入用户的ID,这是不正常,因为ID没有设置过,而且JPA默认的行为是只要指定了主键生成策略,主键就不能设置了,一旦不为空或者0就被认为是已经保存到了数据库中,一旦调用persist()方法就会抛出上面的异常。
为什么会还有第一次的插入用户的信息呢,原因就是我action的作用域设置错误了,我采用了默认单例的设置,对于Action来说必须采用prototype(每次调用创建一个对象)的作用域,修改方法是:在Action上就一个注解@Scope("prototype")