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 ...
随机推荐
- Saltstack批量编译部署nginx(多模块)
最近一直在研究saltstack的同步文件和批量执行命令,随着架构的变大,批量部署的需求也变得明显起来了,我需要用一条命令就部署好nginx和tomcat,并且符合我所有的环境需求,可以直接投入生产环 ...
- HTML文本
1.HTML元素 2.HTML属性 3.HTML文本格式化 4.HTML样式 1.HTML元素 1.什么是HTML元素 HTML 元素指的是从开始标签(start tag)到结束标签(end tag) ...
- 树状数组(Binary Indexed Tree,BIT)
树状数组(Binary Indexed Tree) 前面几篇文章我们分享的都是关于区间求和问题的几种解决方案,同时也介绍了线段树这样的数据结构,我们从中可以体会到合理解决方案带来的便利,对于大部分区间 ...
- 线段树初步__ZERO__.
线段树,顾名思义,就是指一个个线段组成的树. 线段树的定义就是: 线段树是一种二叉搜索树,与区间树相似,它将一个区间划分成一些单元区间,每个单元区间对应线段树中的一个叶结点.使用线段树可以快速的查找某 ...
- 深入理解计算机系统chapter8
进程轮流使用处理器 父进程调用fork来创建一个新的子进程 回收子进程 waitpid/wait 非本地跳转:
- Android Framework 初探
最近工作任务不忙,学习一下Android Framework方面的知识. 一.介绍,是什么 Android的Framework是直接应用之下的一层,叫做应用程序框架层.这一层是核心应用程序所使用的AP ...
- CenOs 部署记录
1.安装APache.即 httpd 2.需要将80端口添加进iptable.外网才能访问.命令:iptables -I INPUT -p TCP --dport 80 -j ACCEPT
- PHP通过URL获取文件大小
function getFileSize($url){ $url = parse_url($url); if($fp = @fsockopen($url['host'],empty($url['por ...
- Python数据分析流程
一.数据分析的步骤: 1.查看数据并提出问题 2.数据清洗 3.代码编写,提取出结果数据,并分析是否有异常数据,修改代码 4.根据数据选择合适的图表进行展示 5.根据图表小组讨论交流获得最终的结果 二 ...
- OC中成员属性 成员变量
比如用property声明一个变量属性 然后我们会为它用懒加载的方式重写get方法 然后我们在使用这个变量的时候,都是用self.itemArray,为什么这样用比较好呢,这是因为self.是对属性的 ...