Hiberante(九) 二级缓存(自己加入缓存)

 

使用二级缓存的步骤
		1、hibernate并没有提供相应的二级缓存的组件,所以需要加入额外的二级缓存包,常用的二级缓存包是ECHcache
		2、在hibernate.cfg.xml中配置开启二级缓存
		<!-- 设置二级缓存为true -->
				<property name="hibernate.cache.use_second_level_cache">true</property>
				<!-- 设置二级缓存所提供的类 -->
				<property name="hibernate.cache.provider_class">net.sf.ehcache.hibernate.EhCacheProvider</property>
				<!-- 在hibernate4.0之后需要设置facotory_class -->
				<property name="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</property>
		3、设置相应的ehcache.xml文件,在这个文件中配置二级缓存的参数,并且将文件在cfg文件中配置
		<!-- 说明ehcache的配置文件路径 -->
				<property name="hibernate.cache.provider_configuration_file_resource_path">ehcache.xml</property>
		4、开启二级缓存
			在xml的配置中设置
			<cache usage="read-only"/>
		5、二级缓存缓存的是对象,它是把所有的对象缓存到内存中,一定注意是基于对象的缓存
		
		6、查询缓存是针对HQL语句的缓存,查询缓存仅仅只会缓存id而不会缓存对象

  

@Test
	public void test01() {
		Session session = null;
		try {
			/**
			 * 此时会发出一条sql取出所有的学生信息
			 */
			session = HibernateUtil.openSession();
			Student stu = (Student)session.load(Student.class, 1);
			System.out.println(stu.getName()+",---");
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			HibernateUtil.close(session);
		}
		try {
			session = HibernateUtil.openSession();
			session.beginTransaction();
			/**
			 *此时session已经关闭了,但是Student在二级缓存中,所以也不会发出SQL语句
			 */
			Student stu = (Student)session.load(Student.class, 1);
			//会报错,因为二级缓存设置为read-only
//			stu.setName("abc");
			System.out.println(stu.getName()+",---");
			session.getTransaction().commit();
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			HibernateUtil.close(session);
		}
	}
	
	
	@Test
	public void test03() {
		Session session = null;
		try {
			/**
			 * 此时会发出一条sql取出所有的学生信息
			 */
			session = HibernateUtil.openSession();
			List<Student> ls = session.createQuery("from Student")
					.setFirstResult(0).setMaxResults(50).list();
			Iterator<Student> stus = ls.iterator();
			for(;stus.hasNext();) {
				Student stu = stus.next();
				System.out.println(stu.getName());
			}
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			HibernateUtil.close(session);
		}
		
		try {
			session = HibernateUtil.openSession();
			session.beginTransaction();
			/**
			 *此时session已经关闭了,但是Student在二级缓存中,所以也不会发出SQL语句
			 */
			Student stu = (Student)session.load(Student.class, 1);
			//会报错,因为二级缓存设置为read-only
//			stu.setName("abc");
			System.out.println(stu.getName()+",---");
			session.getTransaction().commit();
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			HibernateUtil.close(session);
		}
	}
	
	@Test
	public void test04() {
		Session session = null;
		try {
			/**
			 * 此时会发出一条sql取出所有的学生信息
			 */
			session = HibernateUtil.openSession();
			List<Object[]> ls = session.createQuery("select stu.id,stu.name from Student stu")
					.setFirstResult(0).setMaxResults(50).list();
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			HibernateUtil.close(session);
		}
		
		try {
			session = HibernateUtil.openSession();
			session.beginTransaction();
			/**
			 *以上代码仅仅取了id和name,而二级缓存是缓存对象的,所以上一段代码不会将对象加入二级缓存
			 *此时就是发出相应的sql
			 */
			Student stu = (Student)session.load(Student.class, 1);
			//会报错,因为二级缓存设置为read-only
//			stu.setName("abc");
			System.out.println(stu.getName()+",---");
			session.getTransaction().commit();
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			HibernateUtil.close(session);
		}
	}
	
	@Test
	public void test05() {
		Session session = null;
		try {
			/**
			 * 此时会发出一条sql取出所有的学生信息
			 */
			session = HibernateUtil.openSession();
			List<Object[]> ls = session.createQuery("select stu from Student stu")
					.setFirstResult(0).setMaxResults(50).list();
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			HibernateUtil.close(session);
		}
		
		try {
			session = HibernateUtil.openSession();
			/**
			 * 由于学生的对象已经缓存在二级缓存中了,此时再使用iterate来获取对象的时候,首先会通过一条
			 * 取id的语句,然后在获取对象时去二级缓存中,如果发现就不会再发SQL,这样也就解决了N+1问题
			 * 而且内存占用也不多
			 */
			Iterator<Student> stus = session.createQuery("from Student")
					.setFirstResult(0).setMaxResults(50).iterate();
			for(;stus.hasNext();) {
				Student stu = stus.next();
				System.out.println(stu.getName());
			}
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			HibernateUtil.close(session);
		}
	}
	
	@Test
	public void test06() {
		Session session = null;
		try {
			/**
			 * 此时会发出一条sql取出所有的学生信息
			 */
			session = HibernateUtil.openSession();
			List<Object[]> ls = session.createQuery("select stu from Student stu")
					.setFirstResult(0).setMaxResults(50).list();
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			HibernateUtil.close(session);
		}
		
		try {
			session = HibernateUtil.openSession();
			/**
			 * 使用List会发出两条一模一样的sql,此时如果希望不发sql就需要使用查询缓存
			 */
			List<Student> ls = session.createQuery("select stu from Student stu")
					.setFirstResult(0).setMaxResults(50).list();
			Iterator<Student> stus = ls.iterator();
			for(;stus.hasNext();) {
				Student stu = stus.next();
				System.out.println(stu.getName());
			}
		} catch (Exception e) {
			e.printStackTrace();
		} finally {
			HibernateUtil.close(session);
		}
	}
	

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值