一、前言
Hibernate 是一个开放源代码的对象关系映射框架,它对 JDBC 进行了非常轻量级的对象封装,它将 POJO 与数据库表建立映射关系,是一个全自动的 ORM 框架,Hibernate 可以自动生成 SQL 语句,自动执行,使得 Java 程序员可以随心所欲的使用对象编程思维来操纵数据库。
二、数据库连接
2.1 Hibernate 框架引入
- 准备项目所需 jar 包:JAVA_Hibernate_lib.zip。
- 新建项目 JAVA_Hibernate,并导入项目所需 jar 包,如下图所示:
注意复制 jar 包到 lib 文件夹后,要选择所有 jar 包,点击鼠标右键,选择 Build path-->Add to Build path。
- 在 src 源码目录下创建如下包名,完整项目目录结构如下图所示:
- 书写 Hibernate 的主配置文件:hibernate.cfg.xml,存放在 src 目录下。hibernate.cfg.xml 文件如下所示:
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-configuration SYSTEM
"http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd">
<hibernate-configuration>
<session-factory>
<!-- 这个属性使 Hibernate 应用为被选择的数据库生成适当的 SQL -->
<property name="hibernate.dialect">org.hibernate.dialect.DmDialect</property>
<!-- JDBC 驱动程序类 -->
<property name="hibernate.connection.driver_class">dm.jdbc.driver.DmDriver</property>
<!-- Assume test is the database name -->
<!-- 数据库实例的 JDBC URL -->
<property name="hibernate.connection.url">jdbc:dm://localhost:5236</property>
<!-- 数据库用户名 -->
<property name="hibernate.connection.username">SYSDBA</property>
<!-- 数据库密码 -->
<property name="hibernate.connection.password">*****</property>
<!-- 是否显示 SQL -->
<property name="hibernate.show_sql">true</property>
<!-- 是否将 SQL 格式化 -->
<property name="hibernate.format_sql">true</property>
<!-- 是否自动在数据库中生成表 -->
<property name="hibernate.hbm2ddl.auto">update</property>
<!-- 事物自动提交 -->
<property name="hibernate.connection.autocommit">true</property>
<!-- 针对大字段处理 -->
<property name="hibernate.connection.SetBigStringTryClob">true</property>
<!-- List of XML mapping files -->
<!-- 下面是映射的类 -->
<mapping resource="dameng/pojo/BigData.hbm.xml"/>
<mapping resource="dameng/pojo/ProductCategory.hbm.xml"/>
</session-factory>
</hibernate-configuration>
三、开发示例
3.1 基础操作示例
基本操作使用的数据表为示例库中 Product 的 PRODUCT_CATEGORY 表,对应于数据库表添加实体类 ProductCategory,其位置放于 dameng.pojo 下。
- 操作数据库对应的实体类。ProductCategory.java 文件如下所示:
package dameng.pojo;
public class ProductCategory {
private Long productCategoryId;
private String name;
public Long getProductCategoryId() {
return productCategoryId;
}
public void setProductCategoryId(Long productCategoryId) {
this.productCategoryId = productCategoryId;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "ProductCategory [productCategoryId=" + productCategoryId + ", name=" + name + "]";
}
public ProductCategory(Long productCategoryId, String name) {
super();
this.productCategoryId = productCategoryId;
this.name = name;
}
public ProductCategory() {
super();
}
}
- 配置实体类与数据库映射的配置文件。
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="dameng.pojo.ProductCategory" table="PRODUCTION.PRODUCT_CATEGORY">
<id name="productCategoryId" type="java.lang.Long">
<column name="PRODUCT_CATEGORYID" />
<generator class="identity" />
</id>
<property name="name" type="java.lang.String">
<column name="NAME" length="50" not-null="true" />
</property>
</class>
</hibernate-mapping>
- 测试类。TestProduct.java 文件如下所示:
package dameng.test;
import java.util.List;
import org.hibernate.Query;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;
import org.hibernate.classic.Session;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import dameng.pojo.ProductCategory;
public class TestProductCategory {
SessionFactory sf = null;
Session session = null;
@Before
public void before() {
//(1)读取并解析hibernate.cfg.xml配置文件
Configuration config = new Configuration().configure();
//(2)由hibernate.cfg.xml中的<mapping resource="dameng/xx/User.hbm.xml"/>读取并解析映射信息*/
//(3)创建会话SessionFactory对象
sf = config.buildSessionFactory();
//(4)加载Session对象
session = sf.openSession();
}
//测试插入
@Test
public void testInsert() {
ProductCategory pCategory = new ProductCategory(null, "美术");
session.save(pCategory);
}
//测试修改
@Test
public void testUpdate() {
ProductCategory pCategory = null;
pCategory = (ProductCategory)session.get(ProductCategory.class,new Long(4));
System.out.println(pCategory);
pCategory.setName("英语");
System.out.println(pCategory);
session.update(pCategory);
}
//测试全查
@Test
public void testSelectAll() {
String hql = "from ProductCategory";
Query query = session.createQuery(hql);
List<ProductCategory> list = query.list();
for(ProductCategory pCategory:list) {
System.out.println(pCategory);
}
}
//测试删除
@Test
public void testDelete(){
ProductCategory pCategory = null;
pCategory = (ProductCategory)session.get(ProductCategory.class,new Long(5));
if(pCategory!=null) {
session.delete(pCategory);
}
}
@After
public void after() {
if (session != null) {
session.flush();
session.clear();
//(9) 关闭session对象
session.close();
/*
*关闭session工厂,清空资源.由于sessionFactory对象持有整个持久化操作的资源,
*关闭之后,下次在使用又需要创建,太耗内存,通过监听器application servletContext对象,
*当服务器关闭或Web应用被移除时,ServletContext对象跟着销毁,
*当停止进程时,再清理sessionFactory资源
*/
//(10) 关闭sessionFactory对象
sf.close();
}
}
public static void main(String[] args) {
TestProductCategory test = new TestProductCategory();
test.before();
test.testInsert();
test.testUpdate();
test.testSelectAll();
test.testDelete();
test.after();
}
}
- 运行结果截图。运行后控制台输出结果如下图所示:
运行后数据库数据如下图所示:
3.2 大字段操作示例
- 创建需要操作的含大字段类型的数据表。
CREATE TABLE "PRODUCTION"."BIG_DATA"
(
"ID" INT IDENTITY(1, 1) NOT NULL,
"PHOTO" IMAGE,
"DESCRIBE" BLOB,
"TXT" CLOB,
NOT CLUSTER PRIMARY KEY("ID")) STORAGE(ON "BOOKSHOP", CLUSTERBTR) ;
- 在 D 盘根目录下,创建 DM8 特点 .jpg、达梦产品简介 .txt 两个文件,作为大字段存储,如下图所示:
- 操作数据库对应的实体类。BigData.java 文件如下所示:
package dameng.pojo;
public class BigData {
private Long id;
private byte[] photo; //mybatis将 Image 和Blob 映射成byte[]
private byte[] describe;
private String txt; //mybatis 将 Clob 映射成 String
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
public byte[] getPhoto() {
return photo;
}
public void setPhoto(byte[] photo) {
this.photo = photo;
}
public byte[] getDescribe() {
return describe;
}
public void setDescribe(byte[] describe) {
this.describe = describe;
}
public String getTxt() {
return txt;
}
public void setTxt(String txt) {
this.txt = txt;
}
@Override
public String toString() {
return "TestBigDate [id=" + id + ", txt=" + txt + "]";
}
public BigData(Long id, byte[] photo, byte[] describe, String txt) {
super();
this.id = id;
this.photo = photo;
this.describe = describe;
this.txt = txt;
}
public BigData() {
super();
}
}
- 配置实体类与数据库映射的配置文件。
<?xml version="1.0" encoding="utf-8"?>
<!DOCTYPE hibernate-mapping PUBLIC
"-//Hibernate/Hibernate Mapping DTD 3.0//EN"
"http://hibernate.sourceforge.net/hibernate-mapping-3.0.dtd">
<hibernate-mapping>
<class name="dameng.pojo.BigData" table="PRODUCTION.BIG_DATA">
<id name="id" type="java.lang.Long">
<column name="ID" />
<generator class="identity" />
</id>
<property name="photo" type="byte[]">
<column name="PHOTO" />
</property>
<property name="describe" type="byte[]">
<column name="DESCRIBE" />
</property>
<property name="txt" type="java.lang.String">
<column name="TXT" />
</property>
</class>
</hibernate-mapping>
- 测试类。TestBigData.java 文件如下所示:
package dameng.test;
import java.io.BufferedInputStream;
import java.io.BufferedReader;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
import java.io.UnsupportedEncodingException;
import java.util.List;
import org.hibernate.Query;
import org.hibernate.SessionFactory;
import org.hibernate.cfg.Configuration;
import org.hibernate.classic.Session;
import org.junit.After;
import org.junit.Before;
import org.junit.Test;
import dameng.pojo.BigData;
public class TestBigData {
SessionFactory sf = null;
Session session = null;
@Before
public void before() {
//(1)读取并解析 hibernate.cfg.xml 配置文件
Configuration config = new Configuration().configure();
//(2)由 hibernate.cfg.xml 中的<mapping resource="dameng/xx/User.hbm.xml"/>读取并解析映射信息*/
//(3)创建会话 SessionFactory 对象
sf = config.buildSessionFactory();
//(4)加载 Session 对象
session = sf.openSession();
}
//测试插入大字段表
@Test
public void testInsert() {
try {
String filePath = "D:\\DM8特点.jpg";
File file = new File(filePath);
String filePath2 = "D:\\达梦产品简介.txt";
File file2 = new File(filePath2);
InputStream in;
in = new BufferedInputStream(new FileInputStream(file));
byte[] bytes1 = new byte[102400];
byte[] bytes2 = new byte[102400];
in.read(bytes1);
InputStream in2 = new BufferedInputStream(new FileInputStream(file));
in2.read(bytes2);
BufferedReader reader = new BufferedReader(new InputStreamReader
(new FileInputStream(file2),"UTF-8"));
StringBuffer stringBuffer = new StringBuffer("");
String str = null;
while((str = reader.readLine())!=null) {
stringBuffer.append(str);
stringBuffer.append("\n");
}
BigData bigData = new BigData(null,bytes1,bytes2,stringBuffer.toString());
session.save(bigData);
in.close();
in2.close();
reader.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (UnsupportedEncodingException e){
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
//测试查询大字段表
@Test
public void testSelect() {
String hql = "from BigData";
Query query = session.createQuery(hql);
List<BigData> list = query.list();
try {
for(BigData big:list) {
//打印出 id
System.out.println("id = "+big.getId());
//将 photo 列信息 输出到指定路径
FileOutputStream fos = new FileOutputStream("D:/"+big.getId()+"_DM8特点.jpg");
fos.write(big.getPhoto());
//将 describe 列信息 输出到指定路径
FileOutputStream fos2 = new FileOutputStream("D:/"+big.getId()+"_Blob_DM8特点.jpg");
fos2.write(big.getDescribe());
//将 photo 列信息 输出到控制台
System.out.println("txt="+big.getTxt());
fos.close();
fos2.close();
}
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
@After
public void after() {
if (session != null) {
session.flush();
session.clear();
//关闭 session 对象
session.close();
// 关闭 sessionFactory 对象
sf.close();
}
}
public static void main(String[] args) {
TestBigData test = new TestBigData();
test.before();
test.testInsert();
test.testSelect();
test.after();
}
}
- 运行结果截图。运行前如下图所示:
运行后控制台输出 Clob 里保存的大字段文本信息,如下图所示:
运行后读取处理 Image 大字段和 Blob 大字段后的信息,如下图所示:
运行后数据库数据信息如下图所示:
四、常见问题
1. 发生 Error parsing XML: /hibernate.cfg.xml(1) 报错:文档类型声明包含或指向的标记声明格式有误
【问题描述】:使用该链接 http://www.hibernate.org/dtd/hibernate-configuration-3.0.dtd 获得是一个显示 DTD 的简单 HTML 页面,没有获取 DTD 文件本身。
【问题解决】:将该 DTD 保存到本地硬盘,并修改 URL 解决:
2. 插入大字段数据报错 Caused by: dm.jdbc.driver.DMException: 数据大小已超过可支持范围
【问题描述】:排查发现数据库中存储图片的数据类型为 VARBINARY。
【问题解决】:数据类型修改为 IMAGE/BLOB 后解决。
3. Hibernate 连接数据库的方言写法
Hibernate.cfg.xml 配置要求:
驱动名称
<property name="connection.driver_class">dm.jdbc.driver.DmDriver</property>
方言包名称
<property name="dialect">org.hibernate.dialect.DmDialect</property>
五、参考
- 示例代码下载:JAVA_Hibernate.zip 。
- 以上文档内容参考过程中遇到任何问题,可到 达梦技术社区 提问交流。