Hibernate 基础配置及常用功能(三)
本章重点讲述Hibernate对象的三种状态以及如何配置二级缓存
有关Hibernate的三种状态如何相互转换网上都能查到,官方文档描述的也比较详细。这里主要是针对几个重点方法做代码演示。
一、状态转换
package learnhow; import org.hibernate.Session;
import org.hibernate.Transaction;
import org.junit.Test; import util.HibernateUtil;
import model.example.User; public class HibernateTest {
private Session session;
private Transaction transaction; @Test
public void testSaveOrUpdate() {
session = HibernateUtil.getSessionFactory().openSession();
transaction = session.beginTransaction();
// Transient状态:session缓存和数据库中都没有
User user = new User();
user.setUserName("Linda");
session.save(user);
// Persistent状态:save()方法以后,对象获得id并且在session中保留有引用
user = (User) session.get(User.class, 1);
transaction.commit();
session.close();
// Detached状态:session被关闭,数据库和内存中有对象
user.setUserName("John");
session = HibernateUtil.getSessionFactory().openSession();
transaction = session.beginTransaction();
// Persistent状态:此时如果使用save()方法,Hibernate会抛出异常,使用update()方法可以更新数据库数据
session.update(user);
// Transient状态:数据库和session缓存中都删除对象,此时对象有id值,但依然属于T状态
session.delete(user);
transaction.commit();
session.close();
user.setUserName("Bob");
// Transient状态:有id值的T对象为了防止冲突,应该把id值重置
user.setId(0);
session = HibernateUtil.getSessionFactory().openSession();
transaction = session.beginTransaction();
// Persistent状态:重新将对象持久化
session.saveOrUpdate(user);
transaction.commit();
session.close();
}
}
HibernateTest.java
以上代码Hibernate会依次执行插入-更新-删除-插入。
二、延迟加载
package learnhow; import org.hibernate.Session;
import org.hibernate.Transaction;
import org.junit.Test; import util.HibernateUtil;
import model.example.User; public class HibernateTest {
private Session session;
private Transaction transaction; @Test
public void testGetAndLoad() {
session = HibernateUtil.getSessionFactory().openSession();
transaction = session.beginTransaction();
// 立即发出sql语句查询结果
User user1 = (User) session.get(User.class, 2);
transaction.commit();
session.close();
session = HibernateUtil.getSessionFactory().openSession();
transaction = session.beginTransaction();
// 仅返回代理对象,在实际使用的时候才会触发查询条件
User user2 = (User) session.load(User.class, 2);
transaction.commit();
session.close();
// 关闭session以后可以正常使用
System.out.println(user1);
// 关闭session以后查询对象会抛出异常:org.hibernate.LazyInitializationException: could
// not initialize proxy - no Session
System.out.println(user2);
}
}
HibernateTest.java
延迟加载异常是使用Hibernate之中经常遇到的问题。例如,如果在级联条件下设置了fetch = FetchType.LAZY,关闭session以后试图触发查询操作也会抛出异常。
以上代码在4.x和5.x版本中均运行正常,唯一的不同点是在5.x版本中通过get()等方法获取对象的时候不需要强制转型。
三、配置二级缓存
Hibernate通过org.hibernate.testing.cache.CachingRegionFactory对象实现对二级缓存的管理,但是在官方文档中明确注明只适合于测试环境。因此,如果需要在生产环境中使用Hibernate配置二级缓存,官方推荐EHCache。
(1)首先通过Maven引入依赖关系(最后一项依赖是针对二级缓存的,前面都是之前两章用到的包)
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.learnhow</groupId>
<artifactId>Hibernate_Demo</artifactId>
<packaging>war</packaging>
<version>0.0.1-SNAPSHOT</version>
<name>Hibernate_Demo Maven Webapp</name>
<url>http://maven.apache.org</url>
<dependencies>
<!-- 引入新版依赖 -->
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-core</artifactId>
<version>5.0.6.Final</version>
</dependency>
<dependency>
<groupId>javax.servlet</groupId>
<artifactId>servlet-api</artifactId>
<version>2.5</version>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-simple</artifactId>
<version>1.7.13</version>
</dependency>
<dependency>
<groupId>org.javassist</groupId>
<artifactId>javassist</artifactId>
<version>3.20.0-GA</version>
</dependency>
<dependency>
<groupId>javax.transaction</groupId>
<artifactId>jta</artifactId>
<version>1.1</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
<scope>test</scope>
</dependency>
<!-- Hibernate官方推荐的数据库连接池是c3p0 -->
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-c3p0</artifactId>
<version>5.0.6.Final</version>
</dependency>
<!-- 添加二级缓存的实现依赖 -->
<dependency>
<groupId>org.hibernate</groupId>
<artifactId>hibernate-ehcache</artifactId>
<version>5.0.6.Final</version>
</dependency>
</dependencies>
<build>
<finalName>Hibernate5_Demo</finalName>
</build>
</project>
pom.xml
(2)从Hibernate官方文档里找一份ehcache.xml文件,复制到项目的resources目录下
<!--
~ Hibernate, Relational Persistence for Idiomatic Java
~
~ License: GNU Lesser General Public License (LGPL), version 2.1 or later.
~ See the lgpl.txt file in the root directory or <http://www.gnu.org/licenses/lgpl-2.1.html>.
-->
<ehcache> <!-- Sets the path to the directory where cache .data files are created. If the path is a Java System Property it is replaced by
its value in the running VM. The following properties are translated:
user.home - User's home directory
user.dir - User's current working directory
java.io.tmpdir - Default temp file path -->
<diskStore path="./target/tmp"/> <!--Default Cache configuration. These will applied to caches programmatically created through
the CacheManager. The following attributes are required for defaultCache: maxInMemory - Sets the maximum number of objects that will be created in memory
eternal - Sets whether elements are eternal. If eternal, timeouts are ignored and the element
is never expired.
timeToIdleSeconds - Sets the time to idle for an element before it expires. Is only used
if the element is not eternal. Idle time is now - last accessed time
timeToLiveSeconds - Sets the time to live for an element before it expires. Is only used
if the element is not eternal. TTL is now - creation time
overflowToDisk - Sets whether elements can overflow to disk when the in-memory cache
has reached the maxInMemory limit. -->
<defaultCache
maxElementsInMemory="10000"
eternal="false"
timeToIdleSeconds="120"
timeToLiveSeconds="120"
overflowToDisk="true"
/> <!--Predefined caches. Add your cache configuration settings here.
If you do not have a configuration for your cache a WARNING will be issued when the
CacheManager starts The following attributes are required for defaultCache: name - Sets the name of the cache. This is used to identify the cache. It must be unique.
maxInMemory - Sets the maximum number of objects that will be created in memory
eternal - Sets whether elements are eternal. If eternal, timeouts are ignored and the element
is never expired.
timeToIdleSeconds - Sets the time to idle for an element before it expires. Is only used
if the element is not eternal. Idle time is now - last accessed time
timeToLiveSeconds - Sets the time to live for an element before it expires. Is only used
if the element is not eternal. TTL is now - creation time
overflowToDisk - Sets whether elements can overflow to disk when the in-memory cache
has reached the maxInMemory limit. --> <!-- Sample cache named sampleCache1
This cache contains a maximum in memory of 10000 elements, and will expire
an element if it is idle for more than 5 minutes and lives for more than
10 minutes. If there are more than 10000 elements it will overflow to the
disk cache, which in this configuration will go to wherever java.io.tmp is
defined on your system. On a standard Linux system this will be /tmp"
-->
<cache name="sampleCache1"
maxElementsInMemory="10000"
eternal="false"
timeToIdleSeconds="300"
timeToLiveSeconds="600"
overflowToDisk="true"
/> <!-- Sample cache named sampleCache2
This cache contains 1000 elements. Elements will always be held in memory.
They are not expired. -->
<cache name="sampleCache2"
maxElementsInMemory="1000"
eternal="true"
timeToIdleSeconds="0"
timeToLiveSeconds="0"
overflowToDisk="false"
/> --> <!-- Place configuration for your caches following --> </ehcache>
ehcache.xml
(3)配置了二级缓存之后还可以继续配置查询缓存
<?xml version='1.0' encoding='utf-8'?>
<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd"> <hibernate-configuration>
<session-factory>
<!-- 数据库连接配置 -->
<property name="connection.driver_class">com.mysql.jdbc.Driver</property>
<property name="connection.url">jdbc:mysql://localhost:3306/learnhow</property>
<property name="connection.username">root</property>
<property name="connection.password">root</property>
<!-- JDBC连接池,开发环境设置1就可以了 -->
<property name="connection.pool_size">1</property>
<!-- 数据库方言 -->
<property name="dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property>
<!-- session交给hibernate管理 -->
<property name="current_session_context_class">thread</property>
<!-- 是否显示sql语句 -->
<property name="show_sql">true</property>
<!-- 是否对语句格式化输出 -->
<property name="hbm2ddl.auto">update</property>
<!-- 是否对语句格式化输出 -->
<property name="format_sql">true</property>
<!-- 配置二级缓存 -->
<property name="hibernate.cache.use_second_level_cache">true</property>
<property name="hibernate.cache.region.factory_class">org.hibernate.cache.ehcache.EhCacheRegionFactory</property>
<!-- 配置查询缓存,查询缓存需要依赖二级缓存 -->
<property name="hibernate.cache.use_query_cache">true</property> </session-factory>
</hibernate-configuration>
hibernate.cfg.xml
(4)另外还需要在model对象上添加@Cache注解
package model.example; import javax.persistence.Entity;
import javax.persistence.GeneratedValue;
import javax.persistence.Id; import org.hibernate.annotations.Cache;
import org.hibernate.annotations.CacheConcurrencyStrategy; @Entity
@Cache(usage = CacheConcurrencyStrategy.READ_WRITE)
public class User {
private int id;
private String userName; @Id
@GeneratedValue
public int getId() {
return id;
} public void setId(int id) {
this.id = id;
} public String getUserName() {
return userName;
} public void setUserName(String userName) {
this.userName = userName;
} }
User.java
最后使用Junit测试效果:
package learnhow; import java.util.List; import org.hibernate.Session;
import org.hibernate.Transaction;
import org.junit.Test; import model.example.User;
import util.HibernateUtil; public class HibernateTest {
private Session session;
private Transaction transaction; @Test
public void testSecondLevelCache() {
session = HibernateUtil.getSessionFactory().openSession();
transaction = session.beginTransaction();
User user1 = session.get(User.class, 1);
transaction.commit();
session.close();
session = HibernateUtil.getSessionFactory().openSession();
transaction = session.beginTransaction();
// 配置二级缓存以后查询同一个对象只发送一条查询语句
User user2 = session.get(User.class, 1);
transaction.commit();
session.close();
} @Test
public void testQueryCache() {
session = HibernateUtil.getSessionFactory().openSession();
transaction = session.beginTransaction();
String hql = "FROM User";
// 配置了查询缓存以后,第一次查询将结果集保存进缓存中
List<User> users_1 = session.createQuery(hql).setCacheable(true).list();
for (User u : users_1) {
System.out.println(u.getUserName());
}
transaction.commit();
session.close();
session = HibernateUtil.getSessionFactory().openSession();
transaction = session.beginTransaction();
// 第二次查询直接通过缓存获得结果集
List<User> users_2 = session.createQuery(hql).setCacheable(true).list();
for (User u : users_2) {
System.out.println(u.getUserName());
}
transaction.commit();
session.close();
}
}
HibernateTest.java
如果以上测试方法都只触发一条查询语句就代表配置生效了。
(有关Hibernate的基础配置,我能想到的大概就这些,要继续深入学习就得读源码了~呼,好累)
Hibernate 基础配置及常用功能(三)的更多相关文章
- Hibernate 基础配置及常用功能(一)
本来是想等全部框架测试完以后再统一发布的,但是随着测试的一点点增加感觉把需要叙述的东西放在一起终将会是一场灾难.所以还是打算分成几章来描述,其中还包括一些有待解决的问题.短期很难腾出时间来仔细阅读Hi ...
- Hibernate 基础配置及常用功能(二)
本章主要是描述几种经典映射关系,顺带比较Hibernate4.x和Hibernate5.x之间的区别. 一.建立测试工程目录 有关实体类之间的相互映射关系,Hibernate官方文档其实描述的非常详细 ...
- Hibernate学习笔记2.1(Hibernate基础配置)
Hibernate基础配置 1.<property name="hbm2ddl.auto">update</property> 在SessionFactor ...
- Fedora 28 系统基础配置以及常用软件安装方式
实验说明: 很多人说Linux很难用,很难上手,其实不然,倘若不玩游戏,其实很多发行版Linux都可以成为主力系统,就比如本章要讲的 Fedora 28.本章会从镜像来源.系统安装.基础配置和常用软件 ...
- JAVA基础语法:常用功能符以及循环结构和分支结构(转载)
3.JAVA基础语法:常用功能符以及循环结构和分支结构 1.常用功能符 注释 ("文字"是被注释的部分) //文字 单行注释 /文字/ 多行注释 算术运算符 + - * / / 整 ...
- Ansible基础配置与常用模块使用
环境介绍: Ansible服务端IP:192.168.2.215 Ansible客户端IP:192.168.2.216.192.168.2.218.192.168.2.113 一.创建Ansibl ...
- hibernate基础配置
数据库表名和类名 一致 注解:可写可不写: XML:可写可不写: <class name="Student"> 不一致 注解: public class Teache ...
- 3.Hibernate基础配置
1.Hibernate.cfg.xml:hbm2ddl.auto 在SessionFactory创建时,自动检查数据库结构,或者将数据库schema的DDL导出到数据库 <property na ...
- Hibernate学习笔记2.3(Hibernate基础配置)
映射,注释可以放在成员变量上面,也可以放在get方法上面 写在成员变量的话 破坏了java的面向对象思维 直接让hibernate访问内部的私有元素 要是能直接设指不合适哈哈 所以主张写在get方法上 ...
随机推荐
- yii2使用相关记录
#Yii::$app->user 是指yii\web\User这个类 #yii2在命令行下执行 D: cd D:\wnmp\php5 php D:\wnmp\www\yii2\yii test/ ...
- Spark 宏观架构&执行步骤
Spark 使用主从架构,有一个中心协调器和许多分布式worker. 中心协调器被称为driver.Driver 和被称为executor 的大量分布式worker 通信 Driver 运行在它自己的 ...
- SET QUOTED_IDENTIFIER ON和SET ANSI_NULLS ON
distinct是sqlserver的标识符,如果想以distinct为表时,在QUOTED_IDENTIFIER为off的情况下,是不能创建表名为distinct的表的,因为在QUOTED_IDEN ...
- C#中Object和Json之间的转换
//创建一个类,此类有两个方法 using Newtonsoft.Json;namespace System{ public static class JsonExtension { ...
- 一些NSArray,NSDictionary,NSSet相关的算法知识
iOS编程当中的几个集合类:NSArray,NSDictionary,NSSet以及对应的Mutable版本,应该所有人都用过.只是简单使用的话,相信没人会用错,但要做到高效(时间复杂度)精确(业务准 ...
- SQL 随记
内连接 select e.empno as 员工编号, e.ename as 员工名字, d.dname as 部门名字from emp e inner join dept d on e.deptno ...
- 控制反转IOC的依赖注入方式
引言: 项目中遇到关于IOC的一些内容,因为和正常的逻辑代码比较起来,IOC有点反常.因此本文记录IOC的一些基础知识,并附有相应的简单实例,而在实际项目中再复杂的应用也只是在基本应用的基础上扩展而来 ...
- Linux 利用进程打开的文件描述符(/proc)恢复被误删文件
Linux 利用进程打开的文件描述符(/proc)恢复被误删文件 在 windows 上删除文件时,如果文件还在使用中,会提示一个错误:但是在 linux 上删除文件时,无论文件是否在使用中,甚至是还 ...
- Autorelease自动释放池的使用
Autorelease自动释放池的使用 使用ARC开发,只是在编译时,编译器会根据代码结构自动添加了retain.release和autorelease. MRC内存管理原则:谁申请,谁释放 遇到al ...
- html 以及JavaScript 转义
1.之前遇到一个问题,在用js给某个控件赋值时如果含有单引号'或者双引号"有可能导致赋值不成功,如document.getElementById("txtTestBox1" ...