什么是缓存?
缓存说白了,就是应用程序向数据库要数据,然后把一些数据,临时的放在了内存的区域中,第二次再要数据的时候,直接从内存中拿即可。
缓存需要解决的事情:
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 的生命周期保持一致
一级缓存的位置
持久化状态的对象,就是进入了一级缓存中,换句话说,如果一个对象是一个持久化对象,那么这个对象一定在一级缓存中。
二、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两次干吗?
一级缓存的内存结构:
源码:
内存结构图: