Hibernate学习笔记(三) — Hibernate 的一级缓存意义

什么是缓存?

缓存说白了,就是应用程序向数据库要数据,然后把一些数据,临时的放在了内存的区域中,第二次再要数据的时候,直接从内存中拿即可。

缓存需要解决的事情:

1.能把数据放入缓存 2.能把数据从缓存中取出来 3.如果缓存中的数据发生变化,需要把数据同步到数据库中

4.把数据库中的数据同步到缓存中 5.hits命中率低的对象应该及时从缓存中移走

分布式缓存:

为什么会有分布式缓存?

应用程序运行在服务器上,并发访问时,服务器压力过大,分布式缓存就是来分担服务器压力的。

分布式缓存之间的数据是同步的。(比如购物车中的数据都是存在session中)一旦某个服务器挂了,那么操作的数据在其他的服务器的缓存中还可以继续取出来。所以在Tomcat集群的时候,session是先存在了分布式缓存中,在 tomcat 内部集成了memory cache的分布式缓存,就能自动的把 session 同步。

面试,集群时解决 session 问题:就是利用分布式缓存来处理,tomcat可以与memory cache无缝的集成,在tomcat中配置即可。程序员不需要任何干涉。

比较流行的缓存

小型应用:oscache,ehcache

分布式:memory cache,redis,hbase

一、Hibernate的一级缓存

Hibernate的一级缓存,也称为 session 级别的缓存,其生命周期与 session 的生命周期保持一致

一级缓存的位置

Hibernate学习笔记(三) — Hibernate 的一级缓存意义_第1张图片

持久化状态的对象,就是进入了一级缓存中,换句话说,如果一个对象是一个持久化对象,那么这个对象一定在一级缓存中。

二、Session的操作:

session.get():可以把对象放入到一级缓存中,也可以从一级缓存中把对象提取出来(第一次调用放,以后取)

session.save():可以把一个对象放入到一级缓存中

session.evit():可以把一个对象从缓存中清空

session.update():可以把一个对象放入到一级缓存中

session.clear():清空一级缓存中所有的数据

session.close():一级缓存的生命周期结束

测试:

private Session session;
	private Transaction transaction;
	
	@Before
	public void init(){
		session = HibernateUtils.openSession();
		transaction = session.beginTransaction();
	}
	
	@Test
	public void testGet(){
		User user = (User) session.get(User.class, 1);//发sql,把对象放入到一级缓存中
		User user2 = (User) session.get(User.class, 1);//从一级缓存中直接取,不发sql
		//hibernate提供一个统计机制。获取缓存中实体的个数
		int count = session.getStatistics().getEntityCount();
		System.out.println(count);//1,所以get方法把对象放入到缓存
		transaction.commit();
		session.close();
	}
	
	
	@Test
	public void testSave(){
		User user = new User();
		user.setAge(250);
		user.setName("heh");
		session.save(user);
		int count = session.getStatistics().getEntityCount();
		System.out.println(count);//1,所以save方法把对象放入到缓存
		transaction.commit();
		session.close();
	}
	
	@Test
	public void testEvict(){
		User user = (User) session.get(User.class, 1);
		session.evict(user);
		int count = session.getStatistics().getEntityCount();
		System.out.println(count);//0,evit方法把get放入到缓存中的对象,清空了
		transaction.commit();
		session.close();
	}
	
	@Test
	public void testUpdate(){
		User user = (User) session.get(User.class, 1);
		session.evict(user);
		/**
		 * 此处注意,evict方法之后,执行update会发送sql,即使对象不做任何修改
		 * 此处,可以加深session.flush(),进行的对照副本的操作的理解
		 * 注释掉evict,就不会发送update语句
		 */
		session.update(user);
		int count = session.getStatistics().getEntityCount();
		
		System.out.println(count);//1,update方法把对象放入到缓存
		transaction.commit();
		session.close();
	}
	

	@Test
	public void testClear(){
		User user = (User) session.get(User.class, 1);
		session.clear();
		int count = session.getStatistics().getEntityCount();
		System.out.println(count);//0,清空所有
		transaction.commit();
		session.close();
	}

三、一级缓存的真正意义:

从数据库中取出一个班级的所有学生信息,对学生信息进行修改,所有改的操作,都只是针对一级缓存中的数据,只有在 session 的 flush 后,才会与数据库交互。所以不论改多少人的信息,都只是 session.flush之后才会与数据库交互一次。这样就可以提供效率。

而不是get一次,发送一次sql语句,再次get就不发送sql语句,取数据get一次就行,get两次干吗?

Hibernate学习笔记(三) — Hibernate 的一级缓存意义_第2张图片



一级缓存的内存结构

源码:

Hibernate学习笔记(三) — Hibernate 的一级缓存意义_第3张图片

Hibernate学习笔记(三) — Hibernate 的一级缓存意义_第4张图片

内存结构图:

Hibernate学习笔记(三) — Hibernate 的一级缓存意义_第5张图片



你可能感兴趣的:(Hibernate学习笔记(三) — Hibernate 的一级缓存意义)