hibernate mysql 多线程_如何使用Hibernate JPA线程安全进行数据库访问?

本文探讨了如何在多线程环境中使用Hibernate JPA确保数据库访问的安全性。通过分析DbService类的实现,提出了同步方法、实例管理策略以及可能的单例或静态方法设计模式的选择。讨论了add(), delete(), update()和getAll()方法的线程同步需求,并就DbService实例创建的适宜数量提出疑问,引发对应用设计模式的思考。" 117214838,10537790,使用Floyd算法解决超市选址问题,"['C语言', '数据结构', '图论算法', '路径优化', '课程设计']
摘要由CSDN通过智能技术生成

我想知道我需要做什么才能访问数据库线程安全。

这是我的Entity类:

@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类:

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 getAll() {

TypedQuery namedQuery =

em.createNamedQuery("Student.getAll", Student.class);

return namedQuery.getResultList();

}

}

这是与DbService一起使用的类:

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静态所有方法?

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
多线程下,如果使用懒加载(Lazy Loading)加载实体类的关联对象,可能会出现 No session 异常,因为每个线程都有自己的 Session,当一个线程中的 Session 关闭后,另一个线程再去访问同一个实体类的关联对象时,就会出现 No session 异常。 解决这个问题的方法是在每个线程中打开一个 Session,进行操作后再关闭。在 Spring Boot 中,可以使用 @Transactional 注解来管理事务和 Session。 以下是一个完整的示例代码: 实体类: ```java @Entity public class User { @Id private Long id; private String name; @OneToMany(mappedBy = "user", fetch = FetchType.LAZY) private List<Order> orders; // getter 和 setter 略 } @Entity public class Order { @Id private Long id; private String orderNo; @ManyToOne(fetch = FetchType.LAZY) @JoinColumn(name = "user_id") private User user; // getter 和 setter 略 } ``` DAO 层: ```java @Repository public class UserRepository { @Autowired private EntityManager entityManager; public User findById(Long id) { return entityManager.find(User.class, id); } } @Repository public class OrderRepository { @Autowired private EntityManager entityManager; public Order findById(Long id) { return entityManager.find(Order.class, id); } } ``` Service 层: ```java @Service public class UserService { @Autowired private UserRepository userRepository; @Autowired private OrderRepository orderRepository; @Transactional public void test() { User user = userRepository.findById(1L); System.out.println(user.getName()); List<Order> orders = user.getOrders(); // 懒加载 for (Order order : orders) { System.out.println(order.getOrderNo()); } } } ``` 在多线程环境下,可以通过创建一个新的 EntityManager 来避免 No session 异常: ```java @Service public class UserService { @Autowired private UserRepository userRepository; @Autowired private OrderRepository orderRepository; @Transactional public void test() { ExecutorService executorService = Executors.newFixedThreadPool(10); for (int i = 0; i < 10; i++) { executorService.submit(() -> { EntityManager entityManager = null; try { entityManager = JpaUtils.getEntityManager(); User user = userRepository.findById(1L); System.out.println(user.getName()); List<Order> orders = user.getOrders(); // 懒加载 for (Order order : orders) { System.out.println(order.getOrderNo()); } } finally { if (entityManager != null && entityManager.isOpen()) { entityManager.close(); } } }); } executorService.shutdown(); try { executorService.awaitTermination(1, TimeUnit.HOURS); } catch (InterruptedException e) { e.printStackTrace(); } } } ``` 其中,JpaUtils.getEntityManager() 方法用于创建一个新的 EntityManager: ```java public class JpaUtils { private static final EntityManagerFactory entityManagerFactory = Persistence.createEntityManagerFactory("default"); public static EntityManager getEntityManager() { return entityManagerFactory.createEntityManager(); } } ``` 需要在 application.yml 文件中配置数据库连接信息和 JPA 相关的属性: ```yaml spring: datasource: url: jdbc:mysql://localhost:3306/test?useSSL=false&useUnicode=true&characterEncoding=utf-8&serverTimezone=GMT%2B8 username: root password: 123456 driver-class-name: com.mysql.cj.jdbc.Driver jpa: properties: hibernate: dialect: org.hibernate.dialect.MySQL8Dialect show_sql: true format_sql: true hbm2ddl.auto: update open-in-view: false generate-ddl: true hibernate: ddl-auto: update ``` 在以上示例代码中,使用了线程池来模拟多线程请求。在每个线程中,都创建了一个新的 EntityManager,并在 finally 块中关闭。这样可以避免 No session 异常。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值