Spring Boot JPA 常见的那些坑

Spring Boot JPA 常见的那些坑

前言

之前讲解了Spring Boot如何集成JPA和相关特性,虽然JPA使用非常的简单,但是我们在实际的 项目中需要去掌握JPA原理,这样才能更好的解决相关问题。

JPA为什么没有update方法

JPA提供了一个save方法,当主键为空的时候,则执行的为insert语句,当主键不为空的时候,则执行为update方法。

JPA使用Save方法的坑

问题1:需要区分Save方法什么时候执行的insert,什么时候执行的为update语句,切莫误把insert当作了update。

先看下JPA save方法的源码

@Transactional
	@Override
	public <S extends T> S save(S entity)
                 //如果为新对象则执行持久化操作
		if (entityInformation.isNew(entity)) 
		   em.persist(entity);
		   return entity;
                //执行merge方法
	          return em.merge(entity);
        	public boolean isNew(T entity)
                //通过id查询实体对象
		ID id = getId(entity);
		Class<ID> idType = getIdType();
		if (!idType.isPrimitive()) {
			return id == null;
		if (id instanceof Number) {
			return ((Number) id).longValue() == 0L;
		throw new IllegalArgumentException(String.format("Unsupported primitive id type %s!", idType));
复制代码

上述源码可以简单理解,先通过id查询对象是否存在,如果存在则执行更新方法,否则执行新增方法。

新增示例

@RequestMapping("/saveUser")
    public String saveUser(){
        User user =new User();
        user.setName("jpatest");
        user.setAge(100);
        userDao.save(user);
        return "成功";
复制代码

执行结果,我们可以看到输出的为insert语句

Hibernate: 
    insert 
        t_user
        (address, age, email, name) 
    values
        (?, ?, ?, ?)
复制代码

更新示例

@RequestMapping("/updateUser")
    public String updateUser(){
        User user =new User();
        user.setOid(485612);
        user.setName("jpatest122");
        user.setAge(100);
        userDao.save(user);
        return "成功";
    }

第一次执行结果我们可以看到先执行查询语句,然后再执行更新语句。

Hibernate: 
    select
        user0_.oid as oid1_0_0_,
        user0_.address as address2_0_0_,
        user0_.age as age3_0_0_,
        user0_.email as email4_0_0_,
        user0_.name as name5_0_0_ 
        t_user user0_ 
    where
        user0_.oid=?
Hibernate: 
    update
        t_user 
        address=?,
        age=?,
        email=?,
        name=? 
    where
        oid=?
复制代码

如果每次执行update操作都需要先执行查询然后再更新的话,会影响性能。

第二次执行,我们看到只输出了查询语句,没有执行更新操作,这是为什么呢?

Hibernate: 
    select
        user0_.oid as oid1_0_0_,
        user0_.address as address2_0_0_,
        user0_.age as age3_0_0_,
        user0_.email as email4_0_0_,
        user0_.name as name5_0_0_ 
        t_user user0_ 
    where
        user0_.oid=?
复制代码

这里因为session的缓存数据与快照区数据完全相同,没有更新任何属性,所以只会执行查询语句。

问题2:默认情况下,执行update语句会更新实体中的所有字段。

示例代码:

@RequestMapping("/saveOrUpdateUser")
    public String saveOrUpdateUser()
        User user =userDao.findById(485612).orElse(null);
        user.setName("jpatest123sssssxxwwwdds");
        userDao.save(user);
        return "成功";
复制代码

执行结果:我们可以看到,我只想更新用户名字段,但是执行的SQL语句却把实体中的所有字段都进行了更新。

Hibernate: 
    select
        user0_.oid as oid1_0_0_,
        user0_.address as address2_0_0_,
        user0_.age as age3_0_0_,
        user0_.email as email4_0_0_,
        user0_.name as name5_0_0_ 
        t_user user0_ 
    where
        user0_.oid=?
Hibernate: 
    select
        user0_.oid as oid1_0_0_,
        user0_.address as address2_0_0_,
        user0_.age as age3_0_0_,
        user0_.email as email4_0_0_,
        user0_.name as name5_0_0_ 
        t_user user0_ 
    where
        user0_.oid=?
Hibernate: 
    update
        t_user 
        address=?,
        age=?,
        email=?,
        name=? 
    where
        oid=?
复制代码

如果数据库中的字段非常多的话,执行更新操作将严重会影响性能,那么如何解决呢?

解决办法:我们需要在实体类中添加@DynamicUpdate注解,表示动态更新查询。

@DynamicUpdate
public class User
复制代码

添加此注解后,执行SQL语句才是正确的SQl了语句

Hibernate: 
    select
        user0_.oid as oid1_0_0_,
        user0_.address as address2_0_0_,
        user0_.age as age3_0_0_,
        user0_.email as email4_0_0_,
        user0_.name as name5_0_0_ 
        t_user user0_ 
    where
        user0_.oid=?
Hibernate: 
    update
        t_user 
        name=? 
    where
        oid=?
复制代码

问题3:JPA实体状态问题

JPA实体有如下四种状态:

  • 瞬时状态 :瞬时状态的实体就是一个普通的java对象,和持久化上下文无关联,数据库中也没有数据与之对应。
  • 托管状态 :EntityManager进行find或者persist操作返回的对象即处于托管状态,此时该对象已经处于持久化上下文中,因此任何对于该实体的更新都会同步到数据库中。
  • 游离状态 : 当事务提交后,处于托管状态的对象就转变为了游离状态。此时该对象已经不处于持久化上下文中,因此任何对于该对象的修改都不会同步到数据库中。
  • 删除状态 : 当调用EntityManger对实体进行delete后,该实体对象就处于删除状态。其本质也就是一个瞬时状态的对象。

状态之间转换:



我们通过一个实例来讲解:

@Transactional(rollbackFor=Exception.class)
    public void updateUserNameByOid(int oid, String userName)
       //查询数据库数据,将数据存放到缓存区和快照区
        User user = userDao.findById(oid).orElse(null);
        System.out.println("user userName: " + user.getName());
        System.out.println("user age: " + user.getAge());
        System.out.println("userName will be updated to " + userName);
        //查询数据库数据,由于执行update方法没有持久化到数据库
        userDao.modifyNameByOid(userName, 1);
        //由于没有执行更新操作,查询数据都是一样的
        User user1 = userDao.findById(oid).get();
        System.out.println("user1 age: " + user1.getAge());
        System.out.println("age will be updated to " + 18);
        user1.setAge(18);
        //更新年龄字段
        userDao.save(user1);
复制代码

执行结果却出人意料,数据库种用户名字段没有更新,只更新了年龄字段。

Hibernate: 
    select
        user0_.oid as oid1_0_0_,
        user0_.address as address2_0_0_,
        user0_.age as age3_0_0_,
        user0_.email as email4_0_0_,
        user0_.name as name5_0_0_ 
        t_user user0_ 
    where
        user0_.oid=?
user userName: li'si
user age: 1811
userName will be updated to lisi
Hibernate: 
    update
        t_user u 
        u.name = ? 
    where
        u.oid = ?
user1 age: 1811
age will be updated to 18
Hibernate: 
    update