1. Hibernate简介
因为传统的操纵JDBC太繁琐了,首先要加载驱动类,然后从DriverManager中获得连接,接着要写sql、生成语句对象、一堆set操作,最后再执行sql,如果是查询sql的话那么还要对result解包,并且每个Daoimpl都要重复进行这一堆模板代码,因此hibernate应运而生
Hibernate是操纵数据库的框架,不管底层数据库是什么,可以用代码的一致方式去操纵,基本上没有什么影响,换言之,我们在学习hibernate的时候只需要创建java工程就可以了
和struts2框架一样,它也要求我们编写配置文件,其配置文件分为两类,一类是主配置文件(hibernate.cfg.xml)就一个,主要是配置数据源的,还附带一些hibernate的配置,另一类是建立javabean与数据表映射的文件,它的数量与所要操纵的数据表数量有关
hibernate与mybatis的关系,hibernate是全自动的sql,而mybatis是半自动的,也就是说mybatis相对灵活
实体间的关系是hibernate学习困难之处
对于我们手工写的sql而言,IDE在编译的时候是无法获悉sql是否正确的,因为它是字符串,而hibernate将sql的操纵替换成面向对象的方式,使我们可以大胆的使用,不需要担心sql是否因为拼写的问题而出错或是其它相关问题,mybatis中也有类似的设计,在使用mapping文件的时候有两种方式,一种是填写完整的字符串,还有一种是通过Mapper对象来操作的,它使用了动态代理模式,底层还是使用字符串的方式去调用对应的sql的,但是在应用层我们就不需要写字符串来调用了,避免了使用字符串所增加的错误风险
在分层web体系中,我们应该尽量使用对象传递的方式,因为使用离散的变量的话,如果需要增加参数的话,那么涉及到的很多类都需要改变方法签名,但是如果是对象的话,那么就没关系了
在建表的时候,不要将有业务含义的字段来当主键,例如user.username
ORM 即Object Relationship mapping ,对象关系映射,对象可以认为是javabean,关系可以认为是数据库,
对象关系映射文件是hibernate中很重要的文件,将javabean的类名与数据表的表名,javabean的成员变量与数据表的字段名进行了关联
ORM解决的主要问题就是对象与关系的映射
关系模型中,一般通过连接表来表示多对多关联关系的
2. Hibernate整合struts2流程
- 引入struts2需要的jar
- 在web.xml中配置StrutsPrepareAndExecuteFilte过滤器
- 导入hibernate所需的jar,以及mysql连接的jar
- 配置hibernate.cfg.xml文件
- 配置数据表与javabean映射文件
- 使用hibernate提供的api操作数据库就可以了
也就是说在struts2配置好的基础之上,导入一堆hibernate所需的jar以及配置hibernate.cfg.xml和*.hbm.xml文件即可
我们学习的是hibernate 3.2 GA版,具体流程如下:
- 先导入struts2的相关jar包:
- 然后配置web.xml,使struts2能够借助Filter来进行启动
- 启动tomcat
tomcat启动的时候在控制台输出了:[struts-default.xml]、[struts-plugin.xml]、[struts.xml] 一般而言表示struts2成功与servlet对接了,因为struts2借助servlet的Filter来启动的,struts2启动失败的话服务器也就启动失败了
- 访问index.jsp
到这一步表示struts2正式启动了
- 在struts2成功配置好的基础之上开始整合hibernate,先导入hibernate所需要的jar包,hibernate安装包的lib目录下有很多jar包,有一部分是必须的,有一部分是可选的,具体查看_README.txt
因为有个别jar在struts2中已经有了,例如javassist,就不需要加载了
- 在web工程src目录下配置hibernate.cfg.xml,与struts.xml放在同一位置
- 因为connection.url配置了hibernate数据库,所以下一步是创建数据库和表
- 配置数据表,那么就开始配置与之对应的javabean
- 接着配置javabean与数据表的映射文件,放在src目录下
- 接着返回到hibernate.cfg.xml文件中,添加<mapping>元素 【注:很重要,容易忘记】
之后如果想要操纵其它的数据表,那么继续往这里增加mapping元素即可
- 接着使用hibernate提供的api构建SessionFactory
- 搭建传统的分层架构,即Action、Service、Dao
- 配置struts.xml
- 配置前端页面addPerson.jsp、output.jsp
- 测试
提交后,查询数据库以及查看控制台,显示如下:
可以看出,如果数据表不设置自增而是通过hibernate来实现的话,底层就是查两次数据库而已,当然hibernate会使用缓存机制
至此我们hibernate与struts2的整合也就完毕了
3. Hibernate知识补充
- hibernate先被jboss收购,之后jboss又被redhat收购
- hibernate几乎可以屏蔽底层数据库差异,但是各个数据库又都有自己的特征,所以在配置hibernate.cfg.xml的时候,需要通过方言来告知hibernate开发者到底使用了什么数据库,从而方便hibernate更好的处理
- hibernate.cfg.xml中配置show_sql为true之后,通过session操纵数据库的底层sql都会在控制台打印出来,方便开发者开发
- hibernate要求持久化类(javabean)必须提供一个不带参数的默认构造方法,因此底层是通过反射来实例化javabean的
- 一个SessionFactory接口对应一个数据源,应用从sessionfactory中获得session实例,sessionFactory它是线程安全的,并且是重量级的,这意味着不能随意创建或销毁它的实例,如果应用只访问一个数据库,只需要创建一个sessionFactory实例,在dao第一次获取session对象的时候,主动使用了sessionFactoryUtil类,从而使SessionFactory被实例化,也就是说与hibernate的整合是从这一时刻完成的,不同于struts2,如果需要有多个数据源,那么就使用多个sessionFactory即可
- Session接口是hibernate最重要的接口,它也被称为持久化管理器,它提供了持久化相关的操作,如添加、删除、加载、查询对象,它不是线程安全的,因此在设计软件架构时,应该避免多个线程共享一个session实例,session实例是轻量级的,所谓轻量级是指它的创建和销毁不需要消耗太多的资源,这意味着在程序中可以经常创建或销毁session对象(也就是说推荐在方法中创建并使用)
- 通过Configuration对象创建sessionFactory之后,不管对configuration对象做了如何修改都不会影响到sessionFactory,因为它是不可变的,类似string