繁体   English   中英

休眠,多线程和CascadeType.ALL

[英]Hibernate, Multithreading and CascadeType.ALL

在多线程环境中,

这有效

Box box = new Box("B");
Toy t1 = box.addNewToy("t1");
Toy t2 = box.addNewToy("t2");
synchronized (em) {
    em.getTransaction().begin();
    em.persist(t1);
    em.getTransaction().commit();
}
synchronized (em) {
    em.getTransaction().begin();
    em.persist(t2);
    em.getTransaction().commit();
}

但这不是

Box box = new Box("B");
Toy t1 = box.addNewToy("t1");
synchronized (em) {
    em.getTransaction().begin();
    em.persist(t1);
    em.getTransaction().commit();
}
Toy t2 = box.addNewToy("t2");
synchronized (em) {
    em.getTransaction().begin();
    em.persist(t2);
    em.getTransaction().commit();
}

我收到如下错误:“对象引用了一个未保存的瞬态实例”,“具有相同标识符值的另一个对象已与该会话关联”

有任何想法吗?

这是一个重现该问题的最小Maven项目: http : //www.2shared.com/file/bGLmJ6aO/example.html

细节

Java版本“ 1.7.0_17”,休眠4.2.3。最终版,Ubuntu 11.04 natty,SQLite

class Toy {
    @ManyToMany(mappedBy="toys",fetch = FetchType.LAZY, cascade = CascadeType.ALL)
    public List<Box> getBoxes() { return boxes; }

    public void setBoxes(List<Box> boxes) { this.boxes = boxes; }
}

class Box {
    @ManyToMany(fetch = FetchType.LAZY, cascade = CascadeType.ALL)
    public List<Toy> getToys() { return toys; }

    public void setToys(List<Toy> toys) { this.toys = toys; }

    public Toy addNewToy(String name) {
        Toy toy = new Toy();
        toy.setName(name);
        toy.boxes.add(this);
        toys.add(toy);
        return toy;
    }
}

EntityManagerFactory是创建成本高昂的线程安全对象,旨在由所有应用程序线程共享。 它通常在应用程序启动时创建一次。

EntityManager是一种廉价的, 非线程安全的对象,应针对单个业务流程,单个工作单元使用一次,然后丢弃。 ...

参见: http : //docs.jboss.org/hibernate/entitymanager/3.6/reference/en/html_single/#transactions-basics

我不确定自己在做什么,但是如果您正在多个线程中使用相同的静态EntityManager,那就是您的问题。

EntityManagerFactory是您要共享的线程安全对象,但是您应该为正在执行的每个请求/工作单元创建一个新的EntityManager。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM