如何使用 Hibernate JPA 线程安全地访问数据库?
How to make DB access with Hibernate JPA thread-safe?
我想知道我需要做什么才能使对数据库的访问线程安全。
这是我的实体 class:
@Entity
@Table(name = "students")
@NamedQuery(name = "Student.getAll", query = "SELECT s FROM Student s")
public class Student {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private long id;
@Column(length = 32, name = "name")
private String name;
// ... constructor, getters and setters, toString ...
}
这是 DbService class:
public class DbService {
public EntityManager em = Persistence
.createEntityManagerFactory("MyPersistenceUnit")
.createEntityManager();
public Student add(Student student) {
EntityTransaction tx = em.getTransaction();
tx.begin();
Student studentFromDb = em.merge(student);
tx.commit();
return studentFromDb;
}
public void delete(long id) {
EntityTransaction tx = em.getTransaction();
tx.begin();
em.remove(get(id));
tx.commit();
}
public Student get(long id) {
return em.find(Student.class, id);
}
public void update(Student student) {
EntityTransaction tx = em.getTransaction();
tx.begin();
em.merge(student);
tx.commit();
}
public List<Student> getAll() {
TypedQuery<Student> namedQuery =
em.createNamedQuery("Student.getAll", Student.class);
return namedQuery.getResultList();
}
}
这是与 DbService 一起工作的 class:
public class SomeDbWorker implements Runnable {
@Override
public void run() {
DbService service = new DbService();
// do something ...
service.add( ... );
service.delete( ... );
service.getAll();
// ...
}
}
- 制作add()、delete()、update()是否足够和 getAll() 方法是否同步?
- 我可以像在源代码中那样创建多个 DbService 实例吗?还是我只需要创建一个实例?
- 也许我应该使用单例设计模式?或者使用所有方法将 DbService 设为静态?
您不需要使这些方法同步,它们本身就是线程安全的。是的,让 DbService
成为单身人士可能是有意义的。或者,您可以将 em
设为静态。单身人士将是我的首选。您也不希望 em
成为 public。在您使用时将其设为私有。
不需要同步任何东西,
entityManager 不是线程安全的,设计为针对每个工作单元实例化并在之后立即销毁。
相反,工厂的创建成本很高,应该重用
参见 http://docs.oracle.com/javaee/6/tutorial/doc/bnbqw.html "Application-Managed Entity Managers"
和
我想知道我需要做什么才能使对数据库的访问线程安全。
这是我的实体 class:
@Entity
@Table(name = "students")
@NamedQuery(name = "Student.getAll", query = "SELECT s FROM Student s")
public class Student {
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private long id;
@Column(length = 32, name = "name")
private String name;
// ... constructor, getters and setters, toString ...
}
这是 DbService class:
public class DbService {
public EntityManager em = Persistence
.createEntityManagerFactory("MyPersistenceUnit")
.createEntityManager();
public Student add(Student student) {
EntityTransaction tx = em.getTransaction();
tx.begin();
Student studentFromDb = em.merge(student);
tx.commit();
return studentFromDb;
}
public void delete(long id) {
EntityTransaction tx = em.getTransaction();
tx.begin();
em.remove(get(id));
tx.commit();
}
public Student get(long id) {
return em.find(Student.class, id);
}
public void update(Student student) {
EntityTransaction tx = em.getTransaction();
tx.begin();
em.merge(student);
tx.commit();
}
public List<Student> getAll() {
TypedQuery<Student> namedQuery =
em.createNamedQuery("Student.getAll", Student.class);
return namedQuery.getResultList();
}
}
这是与 DbService 一起工作的 class:
public class SomeDbWorker implements Runnable {
@Override
public void run() {
DbService service = new DbService();
// do something ...
service.add( ... );
service.delete( ... );
service.getAll();
// ...
}
}
- 制作add()、delete()、update()是否足够和 getAll() 方法是否同步?
- 我可以像在源代码中那样创建多个 DbService 实例吗?还是我只需要创建一个实例?
- 也许我应该使用单例设计模式?或者使用所有方法将 DbService 设为静态?
您不需要使这些方法同步,它们本身就是线程安全的。是的,让 DbService
成为单身人士可能是有意义的。或者,您可以将 em
设为静态。单身人士将是我的首选。您也不希望 em
成为 public。在您使用时将其设为私有。
不需要同步任何东西,
entityManager 不是线程安全的,设计为针对每个工作单元实例化并在之后立即销毁。
相反,工厂的创建成本很高,应该重用
参见 http://docs.oracle.com/javaee/6/tutorial/doc/bnbqw.html "Application-Managed Entity Managers" 和