JPA状态
一个实体交给JPA维护的时候,这个实体在不同的时期的状态是不一样的
临时(瞬时)
持久(托管)
游离(脱管)
删除状态
脏数据更新
一个持久化状态的数据,如果修改非主键的值在conmmit的时候就会自动发送update语句更新
底层通过缓存来操作
如果一个持久化状态的数据,如果改变主键的值就会报n to n的错误
域对象之间的关系
依赖关系
一个类去依赖另外一个类,比如Service层Dao层
关联关系
一个类和另外一个类他们之间是有关系的
比如员工和部门–多对1
1对多
多对多
1对1
聚合关系
整体和部分关系,整体和部分可以分开存在
组合关系
整体和部分的关系,整体和部分不能分开存在
脏数据更新
一个持久状态对象在事务管理内,如果改变原来的数据(非主键),此时出现脏数据,在事务提交的时候自动发出update语句去修改
第一步:拿到entityManager,开启事务
第二步:
通过entityManager拿到一个对象,那么现在这个对象就是持久化的对象
这个对象会放到一级缓存里面
JPA会为当前这个对象准备一个快照(把这个对象进行了备份)
第三步:提交事务
它会把快照 与 你现在这个对象的数据进行对比
如果相同,就不需要修改,也不会发送SQL(性能就高了)
当不相同的时候,JPA就会认为现在这个数据是脏数据
脏数据它就会在事务提交的时候,把它进行数据库的同步(发送update SQL语句)
二级缓存
属于EntityManagerFactory级别的缓存,他不是自带的,如果要使用二级缓存,相应配置之后,才能使用耳机缓存---通过Encache框架来实现二级缓存
添加persistence.xml配置信息
<!-- 启用二级缓存 -->
<property name="hibernate.cache.use_second_level_cache" value="true" />
<!-- 二级缓存的实现类,文档里面的包名有错的 -->
<property name="hibernate.cache.region.factory_class" value="org.hibernate.cache.ehcache.EhCacheRegionFactory" />
<!-- 启用查询缓存 -->
<property name="hibernate.cache.use_query_cache" value="true" />
在上面添加配置二级缓存扫描的策略
<!-- ALL:所有的实体类都被缓存 -->
<!-- NONE:所有的实体类都不被缓存. -->
<!-- ENABLE_SELECTIVE:标识 @Cacheable(true) 注解的实体类将被缓存 -->
<!-- DISABLE_SELECTIVE:缓存除标识 @Cacheable(false) 以外的所有实体类 -->
<!-- UNSPECIFIED:默认值,JPA 产品默认值将被使用 -->
<shared-cache-mode>ENABLE_SELECTIVE</shared-cache-mode>
二级缓存类配置:命中条件:同一个SessionFactory,不同的entityManager,OID相同
// 一条sql
// @Entity
// @Cacheable(true)
// public class Department
查询缓存
查询缓存:依赖于二级缓存
// 一条sql
// 命中条件:同一个EntityManagerFactory,不同一个entityManager,发出的sql语句必须相同并且查询的条件值也要相同
// 1.如果查询缓存有查询条件,就不要使用查询缓存,因为命中率非常低
// 2.查询缓存里面查询指的是使用jpql,sql进行的查询,就是调用了createQuery方法,find方法不是走查询缓存
// 3.进行domain类的二级缓存的配置