Hibternate框架笔记
Hibernate框架
配置
配置文件:
<?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="hibernate.connection.driver_class">oracle.jdbc.OracleDriver</property>
<property name="hibernate.connection.password">test</property>
<property name="hibernate.connection.url">jdbc:oracle:thin:@localhost:1521:xe</property>
<property name="hibernate.connection.username">test</property>
<property name="hibernate.default_schema">TEST</property>
<property name="hibernate.dialect">org.hibernate.dialect.Oracle10gDialect</property> <property name="hibernate.show_sql">true</property>
<property name="hibernate.format_sql">true</property>
<!-- 自动建表策略 -->
<property name="hibernate.hbm2ddl.auto">update</property> <mapping resource="com/hanqi/model/News.hbm.xml"/>
</session-factory>
</hibernate-configuration>
配置文件详解:
#数据库使用的驱动类
hibernate.connection.driver_class=com.mysql.jdbc.Driver
#数据库连接串
hibernate.connection.url=jdbc:mysql://localhost:3306/db
#数据库连接的用户名
hibernate.connection.username=user
#数据库连接的密码
hibernate.connection.password=password
#数据库使用的方言
hibernate.dialect=net.sf.hibernate.dialect.MySQLDialect
#是否打印SQL语句
hibernate.show_sql=true
javax.persistence.validation.mode=none
Hibernate JDBC 属性
属性名 |
用途 |
hibernate.connection.driver_class |
JDBC driver class |
hibernate.connection.url |
JDBC URL |
hibernate.connection.username |
database user |
hibernate.connection.password |
数据库用户密码 |
hibernate.connection.pool_size |
maximum number of pooled connections |
Hibernate 数据源属性
属性名 |
用途 |
hibernate.connection.datasource |
数据源 JNDI 名字 |
hibernate.jndi.url JNDI |
提供者的 URL(可选) |
hibernate.jndi.class JNDI |
InitialContextFactory 类(可选) |
hibernate.connection.username |
数据库用户(可选) |
hibernate.connection.password |
数据库密码(可选) |
可选的配置属性
有大量属性能用来控制 Hibernate 在运行期的行为。它们都是可选的,并拥有适当的默认值。
属性名 |
用途 |
可选值 ()内为默认 |
hibernate.dialect |
允许 Hibernate 针对特定的关系数据库生成优化的 SQL 的org.hibernate.dialect.Dialect 的类名。 例如:org.hibernate.dialect.MySQLDialect |
|
hibernate.show_sql |
输出所有 SQL 语句到控制台。 |
true|false (false) |
hibernate.format_sql |
在 log 和 console 中打印出更漂亮的 SQL。 |
true|false (false) |
hibernate.default_catalog |
在生成的 SQL 中,将给定的 catalog 附加于非全限定名的表名上 |
|
hibernate.session_factory_name |
org.hibernate.SessionFactory 创建后,将自动使用这个名字绑定到 JNDI 中。 |
|
hibernate.max_fetch_depth |
为单向关联(一对一,多对一)的外连接抓取(outer join fetch)树设置最大深度。 |
0到3 |
hibernate.default_batch_fetch_size |
为 Hibernate 关联的批量抓取设置默认数量。 |
4、8、16 |
hibernate.default_entity_mode |
为由这个 SessionFactory 打开的所有 Session指定默认的实体表现模式。 |
dynamic-map,dom4j,pojo |
hibernate.order_updates |
强制 Hibernate 按照被更新数据的主键,为SQL 更新排序。这么做将减少在高并发系统中事务的死锁。 |
true|false |
hibernate.generate_statistics |
如果开启,Hibernate 将收集有助于性能调节的统计数据。 |
true|false |
hibernate.use_identifier_rollback |
如果开启,在对象被删除时生成的标识属性将被重设为默认值。 |
true|false |
hibernate.use_sql_comments |
如果开启,Hibernate 将在 SQL 中生成有助于调试的注释信息,默认值为 false。 |
true|false (false) |
Hibernate JDBC 和连接(connection)属性、Hibernate 缓存属性、Hibernate 事务属性等主要用于提升性能,并且Hibernate有适当的默认值。入门者可以忽略这些设置,等学到一定阶段,你可以参考官方文档进行适当配置。
注意:
hibernate.hbm2ddl.auto
1.设置为update 只是更新数据不会更改表结构,因此如果你的表结构发生了变化,删除对应的表,或者
删除数据库,从新启动tomcat,重新生成表。
2.设置为validate:加载hibernate时,验证创建数据库表结构,这样 spring在加载之初,如果model层和
数据库表结构不同,就会报错,这样有助于技术运维预先发现问题。
例如:ProductInfoEntity这个实体有property1这个属性,而对应的数据库表product没有property1这个
字段,就会在tomcat启动的时候报错:错误可能如下:Missing column: property1 in wjs.product
3.设置为create:每次加载hibernate,如果数据库中存在表,将所有表删除,然后重新生成表
4.设置为create-drop:加载hibernate时创建,退出时删除表结构 理解:如果一开始数据库没有表,启动
tomcat的时候会生成表,当把tomcat关闭之后生成的表又会消除。即:一开始数据库中有几个表,整个
流程执行结束之后,还是几个表。
Hibernate反向工程
根据表建立实体类:
需要注意如果选择的表没有主键,将会多生成一个作为主键的类。
生成的配置文件:
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<!-- Generated 2017-10-30 11:58:41 by Hibernate Tools 4.3.1 -->
<hibernate-mapping>
<class name="com.hanqi.model.CrmUsers" table="CRM_USERS" optimistic-lock="version">
<id name="id" type="big_decimal">
<column name="ID" precision="22" scale="0" />
<generator class="assigned" />
</id>
<property name="username" type="string">
<column name="USERNAME" length="50" />
</property>
<property name="password" type="string">
<column name="PASSWORD" length="50" />
</property>
<property name="realname" type="string">
<column name="REALNAME" length="50" />
</property>
<property name="tel" type="string">
<column name="TEL" length="20" />
</property>
<property name="email" type="string">
<column name="EMAIL" length="50" />
</property>
<property name="cultural" type="string">
<column name="CULTURAL" length="50" />
</property>
<property name="grade" type="string">
<column name="GRADE" length="20" />
</property>
<property name="scode" type="big_decimal">
<column name="SCODE" precision="22" scale="0" />
</property>
<property name="birthdate" type="string">
<column name="BIRTHDATE" length="20" />
</property>
<property name="address" type="string">
<column name="ADDRESS" length="100" />
</property>
<property name="job" type="string">
<column name="JOB" length="50" />
</property>
<property name="createtime" type="date">
<column name="CREATETIME" length="7" />
</property>
</class>
</hibernate-mapping>
可以在 mapping 中设置包名:
<hibernate-mapping package="">
这样可以省略下面 calss标签中的name属性,会自动寻找。
id标签,主键。
hibernate会根据表中字段的类型自动匹配 model 类中成员变量的类型。
如果:
表中使用了number类型而没有定义长度的话,生成实体类的时候对应的类型会是BigDecimal
表中 number 类型长度与实体类类型对应:
number(1) boolean Boolean
number(2) 2至4之间 byte Byte
number(8) 4至8之间 integer Integer
numbernumber(10)8以上 long Long
<generator class="assigned" />
1、assigned
主键由外部程序负责生成,在 save() 之前必须指定一个。Hibernate不负责维护主键生成。与Hibernate和底层数据库都无关,可以跨数据库。在存储对象前,必须要使用主键的setter方法给主键赋值,至于这个值怎么生成,完全由自己决定,这种方法应该尽量避免。
<id name="id" column="id">
<generator class="assigned" />
</id>
“ud”是自定义的策略名,人为起的名字,后面均用“ud”表示。
特点:可以跨数据库,人为控制主键生成,应尽量避免。
2、increment
由Hibernate从数据库中取出主键的最大值(每个session只取1次),以该值为基础,每次增量为1,在内存中生成主键,不依赖于底层的数据库,因此可以跨数据库。
<id name="id" column="id">
<generator class="increment" />
</id>
Hibernate调用org.hibernate.id.IncrementGenerator类里面的generate()方法,使用select max(idColumnName) from tableName语句获取主键最大值。该方法被声明成了synchronized,所以在一个独立的Java虚拟机内部是没有问题的,然而,在多个JVM同时并发访问数据库select max时就可能取出相同的值,再insert就会发生Dumplicate entry的错误。所以只能有一个Hibernate应用进程访问数据库,否则就可能产生主键冲突,所以不适合多进程并发更新数据库,适合单一进程访问数据库,不能用于群集环境。
官方文档:只有在没有其他进程往同一张表中插入数据时才能使用,在集群下不要使用。
特点:跨数据库,不适合多进程并发更新数据库,适合单一进程访问数据库,不能用于群集环境。
3、hilo
hilo(高低位方式high low)是hibernate中最常用的一种生成方式,需要一张额外的表保存hi的值。保存hi值的表至少有一条记录(只与第一条记录有关),否则会出现错误。可以跨数据库。
<id name="id" column="id">
<generator class="hilo">
<param name="table">hibernate_hilo</param>
<param name="column">next_hi</param>
<param name="max_lo">100</param>
</generator>
</id>
<param name="table">hibernate_hilo</param> 指定保存hi值的表名
<param name="column">next_hi</param> 指定保存hi值的列名
<param name="max_lo">100</param> 指定低位的最大值
也可以省略table和column配置,其默认的表为hibernate_unique_key,列为next_hi
<id name="id" column="id">
<generator class="hilo">
<param name="max_lo">100</param>
</generator>
</id>
hilo生成器生成主键的过程(以hibernate_unique_key表,next_hi列为例):
1. 获得hi值:读取并记录数据库的hibernate_unique_key表中next_hi字段的值,数据库中此字段值加1保存。
2. 获得lo值:从0到max_lo循环取值,差值为1,当值为max_lo值时,重新获取hi值,然后lo值继续从0到max_lo循环。
3. 根据公式 hi * (max_lo + 1) + lo计算生成主键值。
注意:当hi值是0的时候,那么第一个值不是0*(max_lo+1)+0=0,而是lo跳过0从1开始,直接是1、2、3……
那max_lo配置多大合适呢?
这要根据具体情况而定,如果系统一般不重启,而且需要用此表建立大量的主键,可以吧max_lo配置大一点,这样可以减少读取数据表的次数,提高效率;反之,如果服务器经常重启,可以吧max_lo配置小一点,可以避免每次重启主键之间的间隔太大,造成主键值主键不连贯。
特点:跨数据库,hilo算法生成的标志只能在一个数据库中保证唯一。
4、seqhilo
与hilo类似,通过hi/lo算法实现的主键生成机制,只是将hilo中的数据表换成了序列sequence,需要数据库中先创建sequence,适用于支持sequence的数据库,如Oracle。
<id name="id" column="id">
<generator class="seqhilo">
<param name="sequence">hibernate_seq</param>
<param name="max_lo">100</param>
</generator>
</id>
特点:与hilo类似,只能在支持序列的数据库中使用。
5、sequence
采用数据库提供的sequence机制生成主键,需要数据库支持sequence。如oralce、DB、SAP DB、PostgerSQL、McKoi中的sequence。MySQL这种不支持sequence的数据库则不行(可以使用identity)。
<generator class="sequence">
<param name="sequence">hibernate_id</param>
</generator>
<param name="sequence">hibernate_id</param> 指定sequence的名称
Hibernate生成主键时,查找sequence并赋给主键值,主键值由数据库生成,Hibernate不负责维护,使用时必须先创建一个sequence,如果不指定sequence名称,则使用Hibernate默认的sequence,名称为hibernate_sequence,前提要在数据库中创建该sequence。
特点:只能在支持序列的数据库中使用,如Oracle。
6、identity
identity由底层数据库生成标识符。identity是由数据库自己生成的,但这个主键必须设置为自增长,使用identity的前提条件是底层数据库支持自动增长字段类型,如DB2、SQL Server、MySQL、Sybase和HypersonicSQL等,Oracle这类没有自增字段的则不支持。
<id name="id" column="id">
<generator class="identity" />
</id>
例:如果使用MySQL数据库,则主键字段必须设置成auto_increment。
id int(11) primary key auto_increment
特点:只能用在支持自动增长的字段数据库中使用,如MySQL。
7、native
native由hibernate根据使用的数据库自行判断采用identity、hilo、sequence其中一种作为主键生成方式,灵活性很强。如果能支持identity则使用identity,如果支持sequence则使用sequence。
<id name="id" column="id">
<generator class="native" />
</id>
例如MySQL使用identity,Oracle使用sequence
注意:如果Hibernate自动选择sequence或者hilo,则所有的表的主键都会从Hibernate默认的sequence或hilo表中取。并且,有的数据库对于默认情况主键生成测试的支持,效率并不是很高。
使用sequence或hilo时,可以加入参数,指定sequence名称或hi值表名称等,如
<param name="sequence">hibernate_id</param>
特点:根据数据库自动选择,项目中如果用到多个数据库时,可以使用这种方式,使用时需要设置表的自增字段或建立序列,建立表等。
8、uuid
UUID:Universally Unique Identifier,是指在一台机器上生成的数字,它保证对在同一时空中的所有机器都是唯一的。按照开放软件基金会(OSF)制定的标准计算,用到了以太网卡地址、纳秒级时间、芯片ID码和许多可能的数字,标准的UUID格式为:
xxxxxxxx-xxxx-xxxx-xxxxxx-xxxxxxxxxx (8-4-4-4-12)
其中每个 x 是 0-9 或 a-f 范围内的一个十六进制的数字。
<id name="id" column="id">
<generator class="uuid" />
</id>
Hibernate在保存对象时,生成一个UUID字符串作为主键,保证了唯一性,但其并无任何业务逻辑意义,只能作为主键,唯一缺点长度较大,32位(Hibernate将UUID中间的“-”删除了)的字符串,占用存储空间大,但是有两个很重要的优点,Hibernate在维护主键时,不用去数据库查询,从而提高效率,而且它是跨数据库的,以后切换数据库极其方便。
特点:uuid长度大,占用空间大,跨数据库,不用访问数据库就生成主键值,所以效率高且能保证唯一性,移植非常方便,推荐使用。
9、guid
GUID:Globally Unique Identifier全球唯一标识符,也称作 UUID,是一个128位长的数字,用16进制表示。算法的核心思想是结合机器的网卡、当地时间、一个随即数来生成GUID。从理论上讲,如果一台机器每秒产生10000000个GUID,则可以保证(概率意义上)3240年不重复。
<id name="id" column="id">
<generator class="guid" />
</id>
Hibernate在维护主键时,先查询数据库,获得一个uuid字符串,该字符串就是主键值,该值唯一,缺点长度较大,支持数据库有限,优点同uuid,跨数据库,但是仍然需要访问数据库。
注意:长度因数据库不同而不同
MySQL中使用select uuid()语句获得的为36位(包含标准格式的“-”)
Oracle中,使用select rawtohex(sys_guid()) from dual语句获得的为32位(不包含“-”)
特点:需要数据库支持查询uuid,生成时需要查询数据库,效率没有uuid高,推荐使用uuid。
10、foreign
使用另外一个相关联的对象的主键作为该对象主键。主要用于一对一关系中。
<id name="id" column="id">
<generator class="foreign">
<param name="property">user</param>
</generator>
</id>
<one-to-one name="user" class="domain.User" constrained="true" />
该例使用domain.User的主键作为本类映射的主键。
特点:很少使用,大多用在一对一关系中。
11、select
使用触发器生成主键,主要用于早期的数据库主键生成机制,能用到的地方非常少。
12、其他注释方式配置
注释方式与配置文件底层实现方式相同,只是配置的方式换成了注释方式
自动增长,适用于支持自增字段的数据库
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
根据底层数据库自动选择方式,需要底层数据库的设置
如MySQL,会使用自增字段,需要将主键设置成auto_increment。
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
使用表存储生成的主键,可以跨数据库。
每次需要主键值时,查询名为"hibernate_table"的表,查找主键列"gen_pk"值为"2"记录,得到这条记录的"gen_val"值,根据这个值,和allocationSize的值生成主键值。
@Id
@GeneratedValue(strategy = GenerationType.TABLE, generator = "ud")
@TableGenerator(name = "ud",
table = "hibernate_table",
pkColumnName = "gen_pk",
pkColumnValue = "2",
valueColumnName = "gen_val",
initialValue = 2,
allocationSize = 5)
使用序列存储主键值
@Id
@GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "ud")
@SequenceGenerator(name = "ud",
sequenceName = "hibernate_seq",
allocationSize = 1,
initialValue = 2)
13、小结
1、为了保证对象标识符的唯一性与不可变性,应该让Hibernate来为主键赋值,而不是程序。
2、正常使用Hibernate维护主键,最好将主键的setter方法设置成private,从而避免人为或程序修改主键,而使用assigned方式,就不能用private,否则无法给主键赋值。
2、Hibernate中唯一一种最简单通用的主键生成器就是uuid。虽然是个32位难读的长字符串,但是它没有跨数据库的问题,将来切换数据库极其简单方便,推荐使用!
3、自动增长字段类型与序列
数据库 |
自动增长字段 |
序列 |
MySQL |
是 |
|
Oracle |
是 |
|
DB2 |
是 |
是 |
MS SQL Server |
是 |
|
Sybase |
是 |
|
HypersonicSQL |
是 |
|
PostgreSQL |
是 |
|
SAP DB |
是 |
|
HSQLDB |
是 |
|
Infomix |
是 |
4、关于hilo机制注意:
hilo算法生成的标志只能在一个数据库中保证唯一。
当用户为Hibernate自行提供连接,或者Hibernate通过JTA,从应用服务器的数据源获取数据库连接时,无法使用hilo,因为这不能保证hilo单独在新的数据库连接的事务中访问hi值表,这种情况,如果数据库支持序列,可以使用seqhilo。
5、使用identity、native、GenerationType.AUTO等方式生成主键时,只要用到自增字段,数据库表的字段必须设置成自动增加的,否则出错。
6、还有一些方法未列出来,例如uuid.hex,sequence-identity等,这些方法不是很常用,且已被其他方法代替,如uuid.hex,官方文档里建议不使用,而直接使用uuid方法。
7、Hibernate的各版本主键生成策略配置有略微差别,但实现基本相同。如,有的版本默认sequence不指定序列名,则使用名为hibernate_sequence的序列,有的版本则必须指定序列名。
8、还可以自定义主键生成策略,这里暂时不讨论,只讨论官方自带生成策略。
Hibernate使用
测试:
package com.hanqi.test; import java.io.Serializable;
import java.math.BigDecimal;
import java.util.Date; import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
import org.hibernate.cfg.Configuration;
import org.hibernate.service.ServiceRegistry;
import org.junit.After;
import org.junit.Before;
import org.junit.Test; import com.hanqi.model.CrmUsers; public class JUtest { private SessionFactory sessionFactory;
private Session session;
private Transaction transaction; @Before
public void setUp() throws Exception {
System.out.println("开始初始化……");
Configuration configuration=new Configuration().configure("hibernate.cfg.xml");//该属性可以不传值,会自动寻找默认目录下的hibernate.cfg.xml,如果该文件在其他目录中,需要在此指定路径 ServiceRegistry serviceRegistry=new StandardServiceRegistryBuilder().applySettings(configuration.getProperties()).build(); sessionFactory=configuration.buildSessionFactory(serviceRegistry);
session=sessionFactory.openSession();
transaction=session.beginTransaction();
System.out.println("初始化完毕");
} @After
public void tearDown() throws Exception {
System.out.println("释放资源");
transaction.commit(); //提交
session.close();
sessionFactory.close();
System.out.println("关闭");
//以上顺序不能改变
} @Test
public void test() {
//使用session.save方法向表中添加数据
CrmUsers cu=new CrmUsers();
cu.setId(new BigDecimal(3));
cu.setCreatetime(new Date());
Serializable i= session.save(cu);
System.out.println("------------"+i); }
}
实体类三种状态演变:
session.get() 方法查询:
@Test
public void test() {
//查询
CrmUsers cu=(CrmUsers) session.get(CrmUsers.class ,new BigDecimal(3));
System.out.println(cu);
}
先查询在修改,持久化操作:
@Test
public void test() {
//查询
CrmUsers cu=(CrmUsers) session.get(CrmUsers.class ,new BigDecimal(3));
System.out.println(cu);
cu.setRealname("闻道于事");
}
删除,需要先实例化一个实体类,然后再实现时候hibernate会先查出来在删除:
@Test
public void test() {
CrmUsers cu=new CrmUsers();
cu.setId(new BigDecimal(3));
session.delete(cu);
}
hibernate正向工程
创建实体类:
考虑到主键自动生成,所以多加一个没有id的实参构造方法
package com.hanqi.model; import java.util.Date; public class News {
private int id;
private String title;
private String content;
private String author;
private Date redate; public News() {
super();
} public News(String title, String content, String author, Date redate) {
super();
this.title = title;
this.content = content;
this.author = author;
this.redate = redate;
} public News(int id, String title, String content, String author, Date redate) {
super();
this.id = id;
this.title = title;
this.content = content;
this.author = author;
this.redate = redate;
} public int getId() {
return id;
} public void setId(int id) {
this.id = id;
} public String getTitle() {
return title;
} public void setTitle(String title) {
this.title = title;
} public String getContent() {
return content;
} public void setContent(String content) {
this.content = content;
} public String getAuthor() {
return author;
} public void setAuthor(String author) {
this.author = author;
} public Date getRedate() {
return redate;
} public void setRedate(Date redate) {
this.redate = redate;
} @Override
public String toString() {
return "News [id=" + id + ", title=" + title + ", content=" + content + ", author=" + author + ", redate="
+ redate + "]";
} }
创建映射文件:
建立hibernate总配置文件:
如果还没有建立连接,需要再次创建连接:
生成配置文件:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE hibernate-configuration PUBLIC
"-//Hibernate/Hibernate Configuration DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
<property name="hibernate.connection.driver_class">oracle.jdbc.OracleDriver</property>
<property name="hibernate.connection.password">test</property>
<property name="hibernate.connection.url">jdbc:oracle:thin:@localhost:1521:xe</property>
<property name="hibernate.connection.username">test</property>
<property name="hibernate.default_schema">TEST</property>
<property name="hibernate.dialect">org.hibernate.dialect.Oracle10gDialect</property> <!-- 在控制台打印SQL语句 -->
<property name="hibernate.show_sql">true</property>
<property name="hibernate.format_sql">true</property>
<!-- 自动建表策略 -->
<property name="hibernate.hbm2ddl.auto">update</property> <mapping resource="com/hanqi/model/News.hbm.xml"/>
</session-factory>
</hibernate-configuration>
然后新加一个测试,test中不需要内容直接运行,hibernate就会在数据库中创建表:
package com.hanqi.util; import static org.junit.Assert.*; import org.hibernate.Session;
import org.hibernate.SessionFactory;
import org.hibernate.Transaction;
import org.hibernate.boot.registry.StandardServiceRegistryBuilder;
import org.hibernate.cfg.Configuration;
import org.hibernate.service.ServiceRegistry;
import org.junit.After;
import org.junit.Before;
import org.junit.Test; public class JUTest { private SessionFactory sessionFactory;
private Session session;
private Transaction transaction; @Before
public void setUp() throws Exception {
System.out.println("开始初始化……");
Configuration configuration=new Configuration().configure("hibernate.cfg.xml"); ServiceRegistry serviceRegistry=new StandardServiceRegistryBuilder().applySettings(configuration.getProperties()).build(); sessionFactory=configuration.buildSessionFactory(serviceRegistry);
session=sessionFactory.openSession();
transaction=session.beginTransaction();
System.out.println("初始化完毕");
} @After
public void tearDown() throws Exception {
System.out.println("释放资源");
transaction.commit(); //提交
session.close();
sessionFactory.close();
} @Test
public void test() { } }
在控制台看到这个就表示数据库更新成功:
HQL语句:
HQL是Hibernate Query Language的缩写,提供更加丰富灵活、更为强大的查询能力;HQL更接近SQL语句查询语法。
需要注意的是在HQL语句中,form后面跟的不是表名,而是实体类
//批量添加数据
// for(int i=1;i<11;i++) {
// News news = new News("标题"+i, "内容"+i, "作者"+i, new Date());
// session.save(news);
// } //全表查询
String hql = "from News"; Query query = session.createQuery(hql); List<News> list = query.list();
for(News n : list) {
System.out.println(n);
}
条件查询:
参数名称绑定
参数位置绑定
setParameter()方法
setParameterList(集合或者数组)方法
setProperties()方法
参数名称绑定和参数位置绑定:
//条件查询
//参数位置绑定 setParameter()方法
String hql = "from News as n where n.id =?"; Query query = session.createQuery(hql)
.setParameter(0, 10); List<News> list = query.list();
for(News n : list) {
System.out.println(n);
}
//参数名称绑定 setParameterList(集合或者数组)方法
// String hql = "from News as n where n.id in :ids";
//
// Query query = session.createQuery(hql)
// .setParameterList("ids", new Object[]{23, 28});
//
// List<News> list = query.list();
// for(News n : list) {
// System.out.println(n);
// }
hibernate框架一对一和一对多
首先用反向工程把两个具有主外键关系的表生成实体类:
生成的配置文件:
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<!-- Generated 2017-10-31 14:20:12 by Hibernate Tools 4.3.1.Final -->
<hibernate-mapping>
<class name="com.hanqi.model.Emp" table="EMP" optimistic-lock="version" lazy="false">
<id name="empno" type="int">
<column name="EMPNO" precision="8" scale="0" />
<generator class="sequence">
<param name="sequence">sq_emp</param>
</generator>
</id>
<many-to-one name="dept" class="com.hanqi.model.Dept" fetch="select">
<column name="DEPTNO" precision="8" scale="0" />
</many-to-one>
<property name="ename" type="string">
<column name="ENAME" length="320" />
</property>
<property name="birthday" type="date">
<column name="BIRTHDAY" length="7" />
</property>
<property name="sals" type="java.lang.Integer">
<column name="SALS" precision="7" scale="0" />
</property>
<property name="sex" type="java.lang.Byte">
<column name="SEX" precision="2" scale="0" />
</property>
<property name="hiredate" type="date">
<column name="HIREDATE" length="7" />
</property>
<property name="level" type="string">
<formula>(select (CASE WHEN e.sals <![CDATA[<]]> 5000
THEN '普通' ELSE '高级' END)
from emp e where e.empno=empno)</formula>
</property>
</class>
</hibernate-mapping>
<?xml version="1.0"?>
<!DOCTYPE hibernate-mapping PUBLIC "-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://www.hibernate.org/dtd/hibernate-mapping-3.0.dtd">
<!-- Generated 2017-10-31 14:20:12 by Hibernate Tools 4.3.1.Final -->
<hibernate-mapping>
<class name="com.hanqi.model.Dept" table="DEPT" optimistic-lock="version">
<id name="deptno" type="int">
<column name="DEPTNO" precision="8" scale="0" />
<generator class="sequence">
<param name="sequence">sq_dept</param>
</generator>
</id>
<property name="dname" type="string">
<column name="DNAME" length="320" />
</property>
<property name="loc" type="string">
<column name="LOC" length="320" />
</property>
<set name="emps" table="EMP" inverse="true" lazy="false" fetch="select">
<key>
<column name="DEPTNO" precision="8" scale="0" />
</key>
<one-to-many class="com.hanqi.model.Emp" />
</set>
</class>
</hibernate-mapping>
可以看到两个配置文件中:
optimistic-lock="version" : 乐观锁
<many-to-one name="Dept" class="com.hanqi.model.Dept" fetch="select">
<column name="DEPTNO" precision="22" scale="0" />
</many-to-one> 多对一
<set name="Emp" table="Emp" inverse="true" lazy="true" fetch="select">
<key>
<column name="DEPTNO" precision="22" scale="0" />
</key>
<one-to-many class="com.hanqi.model.Emp" />
</set> 一对多
通过session.get方法和session.load方法进行查询:
@Test
public void test() {
//通过get方法查询:
Dept pd=(Dept)session.get(Dept.class,1);
System.out.println("get:"+pd); //通过load方法查询:
Dept pd1=(Dept)session.load(Dept.class,1);
System.out.println("get:"+pd1);
}
两种方法的却别在于:
get:
使用get方法查询一条没有的记录会返回null
load:
使用load方法差一条没有的记录会报错
load方法支持延迟加载
进行查询,使用 formula 标签。
比如查询员工工资等级
在实体类中加一个成员变量 level,添加get/set方法
在配置文件中加:
<property name="level" type="string">
<formula>(select (CASE WHEN e.sals <![CDATA[<]]> 5000
THEN '普通' ELSE '高级' END)
from emp e where e.empno=empno)</formula>
</property>
然后就可以进行查询了:
@Test
public void test() {
//通过load方法查询:
Emp e=(Emp)session.load(Emp.class,6);
System.out.println(e.getEname()+e.getLevel());
}
一对多查询:
@Test
public void test() {
//通过部门主键查部门的员工
String hql = "from Emp as e where e.dept.deptno=1";
List<?> list = session.createQuery(hql).list();
for(Object e : list) {
System.out.println(e);
}
}
@Test
public void test() {
//模糊查询
String hql = "select new map(e.ename as en, e.sals as es) from Emp as e "
+ "where e.dept.dname like '%发%'";
List<?> list = session.createQuery(hql).list();
for(Object e : list) {
System.out.println(e);
}
}
一对多的几种写法:
使用构造方法,需要需要有对应的构造犯法,输出输出所有属性
//使用指定的构造方法
String hql3 = "select new Emp(e.ename as en, e.sals as es) from Emp as e "
+ "where e.dept.dname like '%发%'";
List<Emp> list3 = session.createQuery(hql3).list();
for(Object e1 : list3) {
System.out.println(e1);
}
使用map方式:
//使用map方式
String hql1 = "select new map(e.ename as en, e.sals as es) from Emp as e "
+ "where e.dept.dname like '%发%'";
List<?> list1 = session.createQuery(hql1).list();
for(Object e : list1) {
System.out.println(e);
}
使用list方式:
// //使用集合方式
String hql2 = "select new list(e.ename as en, e.sals as es) from Emp as e "
+ "where e.dept.dname like '%发%'";
List<?> list2 = session.createQuery(hql2).list();
for(Object e : list2) {
System.out.println(e);
}
悲观锁和乐观锁:
悲观锁(Pessimistic Lock), 顾名思义,就是很悲观,每次去拿数据的时候都认为别人会修改,所以每次在拿数据的时候都会上锁,这样别人想拿这个数据就会block直到它拿到锁。传统的关系型数据库里边就用到了很多这种锁机制,比如行锁,表锁等,读锁,写锁等,都是在做操作之前先上锁。
乐观锁(Optimistic Lock), 顾名思义,就是很乐观,每次去拿数据的时候都认为别人不会修改,所以不会上锁,但是在更新的时候会判断一下在此期间别人有没有去更新这个数据,可以使用版本号等机制。乐观锁适用于多读的应用类型,这样可以提高吞吐量,像数据库如果提供类似于write_condition机制的其实都是提供的乐观锁。
两种锁各有优缺点,不可认为一种好于另一种,像乐观锁适用于写比较少的情况下,即冲突真的很少发生的时候,这样可以省去了锁的开销,加大了系统的整个吞吐量。但如果经常产生冲突,上层应用会不断的进行retry,这样反倒是降低了性能,所以这种情况下用悲观锁就比较合适。
Hibternate框架笔记的更多相关文章
- MyBatis 框架笔记
Mybatis 框架笔记 ------技术源于热爱! 获取更多内容请关注小编的个人微信公众平台 1 Mybatis入门 1.1 单独使用jdbc编程问题总结 1.1.1 jd ...
- Java基础知识强化之集合框架笔记76:ConcurrentHashMap之 ConcurrentHashMap简介
1. ConcurrentHashMap简介: ConcurrentHashMap是一个线程安全的Hash Table,它的主要功能是提供了一组和Hashtable功能相同但是线程安全的方法.Conc ...
- 深度学习Keras框架笔记之AutoEncoder类
深度学习Keras框架笔记之AutoEncoder类使用笔记 keras.layers.core.AutoEncoder(encoder, decoder,output_reconstruction= ...
- 深度学习Keras框架笔记之TimeDistributedDense类
深度学习Keras框架笔记之TimeDistributedDense类使用方法笔记 例: keras.layers.core.TimeDistributedDense(output_dim,init= ...
- 深度学习Keras框架笔记之Dense类(标准的一维全连接层)
深度学习Keras框架笔记之Dense类(标准的一维全连接层) 例: keras.layers.core.Dense(output_dim,init='glorot_uniform', activat ...
- Java EE数据持久化框架笔记 • 【目录】
章节 内容 实践练习 Java EE数据持久化框架作业目录(作业笔记) 第1章 Java EE数据持久化框架笔记 • [第1章 MyBatis入门] 第2章 Java EE数据持久化框架笔记 • [第 ...
- 阅读Facebook POP框架 笔记(一)
在这一系列文章里,我主要会将自己阅读第三方代码的经历记录下来,尝试独立分析解剖一个框架.之前也阅读过一些第三方代码,但是实际上来说对自己的成长并没有太大的帮助,因为阅读的不细致,没有领会到代码的精髓. ...
- [Tensorflow实战Google深度学习框架]笔记4
本系列为Tensorflow实战Google深度学习框架知识笔记,仅为博主看书过程中觉得较为重要的知识点,简单摘要下来,内容较为零散,请见谅. 2017-11-06 [第五章] MNIST数字识别问题 ...
- Python Django框架笔记(三):django工作方式简单说明和创建用户界面
(一) 说明 简单说明下django的工作方式,并举2个例子. (二) Django工作方式 假定我们有下面这些文件 ,这里在前2篇的基础上增加了 templates目录(存放html文件) 和s ...
随机推荐
- Shiro第一篇【Shiro的基础知识、回顾URL拦截】
Shiro基础知识 在学习Shiro这个框架之前,首先我们要先了解Shiro需要的基础知识:权限管理 什么是权限管理? 只要有用户参与的系统一般都要有权限管理,权限管理实现对用户访问系统的控制,按照安 ...
- 基于jquery开发的UI框架整理分析
根据调查得知,现在市场中的UI框架差不多40个左右,不知大家都习惯性的用哪个框架,现在市场中有几款UI框架稍微的成熟一些,也是大家比较喜欢的一种UI框架,那应该是jQuery,有部分UI框架都是根据j ...
- Django实现内容缓存
1.缓存的简介 在动态网站中,用户所有的请求,服务器都会去数据库中进行相应的增,删,查,改,渲染模板,执行业务逻辑,最后生成用户看到的页面. 当一个网站的用户访问量很大的时候,每一次的的后台操作,都会 ...
- day14<常见对象+>
常见对象(正则表达式的概述和简单使用) 常见对象(字符类演示) 常见对象(预定义字符类演示) 常见对象(数量词) 常见对象(正则表达式的分割功能) 常见对象(把给定字符串中的数字排序) 常见对象(正则 ...
- 如何快速成长?我的java之路!
由于一些外部的原因,我不得不从自己熟悉的php领域,转战到java战场.我个人觉得还是有些心得吧,不管怎么样,或多或少可能都会有那么些经历的人,和你一起走在这世上!尽管你不知道TA是谁. 其实,转换一 ...
- Python 接口测试(四)
五:python数列化和反序列化 把python的对象编码转换为json格式的字符串,反序列化可以理解为:把json格式 字符串解码为python数据对象.在python的标准库中,专门提供了json ...
- 线段树专题—ZOJ1610 Count the Colors(涂区间,直接tag标记)
Painting some colored segments on a line, some previously painted segments may be covered by some th ...
- MySQL之最基本命令
前言:以下是数据库最基础最常用的命令,特别适用初学者练习,希望通过不断练习这些命令来熟练操作.巩固基础,因为只有不断地练习才能将知识真正变成自己的东西. 快速查看以下内容: 操作 命令 创建数据库 C ...
- 前端基础之初识 HTML
HTML HTML(Hypertext Markup Language)即超文本标记语言,是WWW的描述语言.设计HTML语言的目的是为了能把存放在一台电脑中的文本或图形与另一台电脑中的文本或图形方便 ...
- 基于SSM之Mybatis接口实现增删改查(CRUD)功能
国庆已过,要安心的学习了. SSM框架以前做过基本的了解,相比于ssh它更为优秀. 现基于JAVA应用程序用Mybatis接口简单的实现CRUD功能: 基本结构: (PS:其实这个就是用的Mapper ...