ORM(Object/ReLationshop Mapping):对象关系映射,尽量少写与底层相关的sql语句,便于维护。Hibernate为ORM框架
1)Hibernate简介
Hibernate是java领域的一款开源的ORM框架技术
Hibernate对JDBC进行了非常轻量级的对象封装
2)Hibernate在我们的应用程序中充当着什么样的角色呢?
举例,如下图所示:我们要开发不同的系统
持久化层 --- 处于业务逻辑层和数据库之间的一个角色
作用 -- 把程序中生成的对象持久化到数据库中,换句话说,就是把这些对象通过对象关系映射保存到数据库的表中
3)其他主流的ORM框架技术
①MyBtis : 前身就是著名的iBatis
②Toplink:后被Oracle收购,并重新包装为Oracle AS TopLink
③EJB:本身是JAVAEE的规范
JUnit测试
@Test 测试方法
@Before 初始化方法
@After 释放资源
执行顺序:@Before --> @Test --> @After
hibernate中获得数据库连接
//创建配置对象
Configuration config = new Configuration().configure();
//创建服务注册对象
ServiceRegistry serviceRegistery = new ServiceRegistryBuilder().applySettings(config.getProperties()).buildServiceRegistry();
//创建会话工厂
SessionFactory sessionFactory = config.buildSessionFactory(serviceRegistery);
//创建会话对象
Session session = sessionFactory.openSession();
//开启事务
Transaction transaction = session.beginTransaction();
hibernate常用配置
hibernate前缀可以省略。
hibernate执行流程:
Configuration读取配置文档hibernate.cfg.xml文档 读取文档为了创建sessionfactory对象,创建session工厂对象时读取映射文件一般之创建一个 然后获取session(数据库连接对象)执行session方法之前必须开启事物提交事物后session 事物transaction
Transaction事务简介:
- hibernate对数据库的操作都是封装在事务当中的,并且默认是非自动提交的方式。所以用session保存对象时,如果不开启事务并且没有手工提交事务,对象并不会真正保存在数据库中。
- 如果你想让hibernate像jdbc那样自动提交事务,必须调用session对象的doWork()方法,获得jdbc的connection后,设置其为自动提交事务模式(注意:通常并不推荐这样做)
综上,我们必须开启事务。
使用Session的doWork()方法提交事务(需要重写execute()方法)。
需要注意的是:在使用save()方法后并不会真正输出sql语句,需要调用flush()强制输出sql语句才可以。然后因为采用了自动提交方式(setAutoCommit(true)),数据才真正保存在数据库。
openSession与getCurrentSession区别:
1:openSession 每次使用都是打开一个新的session,使用完需要调用close方法关闭session;不关闭session会导致连接池溢出。
2:getCurrentSession 是获取当前session对象,连续使用多次时(即使关闭了上一个session,再次获取一个 session,获取到的仍然是同一个session对象),得到的session都是同一个对象;
另外,事务提交或回滚后,会自动关闭session对象。
3:一般在实际开发中,往往使用getCurrentSession多,因为一般是处理同一个事务,所以在一般情况下比较少使用openSession;
hibernate-mapping标签:
class标签:
batch-size:抓取策略,表示一次可以抓取多少记录。
id标签:
(子标签表示主键生成策略)
主键生成策略:
单一主键主键生成策略
native:mysql下为auto_increment,从1开始
assigned:程序中手工赋值,注意对象在实例化时id属性自动初始化为0了
hibernate映射类型、java数据类型、sql数据类型的对应关系
即在xxx.hbm.xml中<id>或者<property>的type属性,一般用hibernate映射类型或java数据类型
1.基本数据类型
对象类型:
组件属性<component name="某类的对象" class="刚才的某类">
get与load的区别:
1.get调用后直接发出sql语句获取对象,load调用后返回一个存有id(主键)的代理对象,当程序中需要使用这个对象的其他属性时才调用sql语句获取对象
2.当找不到相应数据时,get返回null,load抛出异常org.hibernate.ObjectNotFoundException
总结:
1.什么是ORM?为什么使用Hibernate?
对象关系映射;为了少写和底层数据库相关的sql语句,方便程序的维护、修改,提高跨平台性和可扩展性。Hibernate是Java领域内的一款技术成熟稳定的ORM框架
2.Hibernate开发的基本步骤?
(1)编写配置文档hibernate.cfg.xml
(2)编写实体类。注意:每一个实体类都要与数据库中的一张表一一对应,实体类的编写要遵循JavaBean的要求。
(3)生成对应实体类的映射文件并添加到配置文档中
(4)调用Hibernate API进行测试
3.什么是session?
类似于JDBC里面的connection对象。调用session操作数据库,实际上就是调用connection的各种API函数来实现的。
4.openSession与getCurrentSession的区别?
前者每次都是创建新的session对象,而后者使用单例模式,每次创建都是相同的对象。openSession在使用完毕后需要显式地关闭,而getCurrentSession在事务提交之后会自动关闭。
5.单表操作有哪些常用的方法?
增删改查对应使用session当中的save、delete、update、get/load方法
6.单表操作,查询一条记录时,get和load的区别?
get在使用的时候立即发送sql语句,并且获得的是实体类的对象类型,而load只有在使用具体对象的非主属性的时候才会发送sql语句,而且返回的是一个代理对象。
单向多对一关联
·多对一的关系和关系数据库中的外键参照关系最匹配,即在己方的表中的一个外键参照另一个表的主键
·通过在多方持有一方的引用实现,需要在“多”端的一端使用<many to one> 配置
<many-to-one name="grade" class="com.imooc.entity.Grade" column="gid" cascade="all"></many-to-one>
</class>
inverse属性;
1、一方的hbm.xml文件的<set>节点的inverse属性指定了关联关系的控制方向,默认由one方来维护;
2、关联关系中,inverse="false"则为主动方,由主动方负责维护关联关系;
3、在一对多关联中,设置one方的inverse为true,这将有助于性能的改善。
cascade属性:
1、当设置了cascade属性不为none时,Hibernate会自动持久化所关联的对象;
2、cascade属性的设置会带来性能上的变动,需要谨慎设置;
————————————————————————————
属性值 含义和作用
all 对所有操作进行级联操作
save-update 执行保存和更新操作时进行级联操作
delete 执行删除时进行级联操作
none 对所有操作不进行级联操作
————————————————
说明:
1.在向Grade添加Student后,保存Grade对象时,能不能不需要显示的保存Student对象,答案是可以的,可以使用级联操作:在Grade的映射文件中的<set>节点添加cascade属性不为none,为"save-update";
2.在向Student添加Grade后,保存Student对象时,能不能不需要显示的保存Grade对象,答案是可以的,可以使用级联操作:在Student的映射文件中的<many-to-one>节点添加cascade属性不为none,为"save-update".
单向一对多:
单向多对一:
HQL定义:
1、HQL:Hibernate Query Lauguage,Hibernate查询语言;
2、HQL是面向对象的查询语言(即以面向对象的角度进行查询),它查询的主体是映射配置的持久化类及其属性,与SQL查询主体是数据库表是完全不同的,除此之外,从数据库查询方面的功能上来讲,SQL能实现的数据库查询功能,HQL几乎可以同样实现。实际上作为一个ORM对象关系映射框架,Hibernate框架会将编写好的HQL语句解析成SQL语句来完成最终的数据库查询操作。
3、HQL提供了丰富灵活的查询特性,是Hibernate官方推荐查询方式。
Query接口简介
1.org.hibernate.Query接口定义有执行查询的方法(该方法完成HQL语句的解析与执行过程,并返回查询的结果。就像SQL语句没有jdbc接口,它也就是普通的字符串变量,HQL语句也一样,编写好的HQL语句也就是一个普通的字符串变量,而hibernate框架就负责解析HQL语句,然后根据配置信息生成相应的SQL语句来执行数据库的查询操作,那么完成这个过程依靠的就是Query接口);
2.Query接口支持方法链编程风格,使得程序代码更为简洁(方法链编程:调用方法后,返回的结果依然是调用这个方法的对象,可以在调用方法后直接调用该对象的其他方法,这样可以使用一个程序语句完成多个方法的调用与执行。在Query接口中,方法链编程使用最多的场景是查询参数的动态设置,特别是多个参数的设置)
3.Query实例的创建:
a.Session的createQuery()方法创建Query实例
b.createQuery方在调用时需要传递一个参数(即要查询的HQL语句),createQuery(hql)
4.Query执行查询
a.Query接口的list()方法执行HQL查询
b.list()方法返回结果数据类型为java.util.List,List集合中存放符合查询条件的持久化对象
from子句
from子句是HQL语句的最简形式,换句话说,HQL语句只需要from子句就可以执行查询了,这是为什么呢?
这是因为from子句指定了HQL语句查询的主体----映射配置的持久化类及其属性,当我们编写的HQL语句只有from子句的时候,HQL框架默认就是查询该持久化类的所有实例以及该持久化类映射配置的信息,当HQL框架将HQL解析成SQL时,就会查询该持久化类映射的数据表中的所有映射字段信息,并将返回的查询结果封装成该持久化类的list集合
distinct的使用
select distinct s.star from Seller s
使用distinct去除s.star中的重复元素