嗯,,我就先用这个标题来作为我学习Hibernate的起始记录贴名字。
我开始学习Hibernate已经好几天了,然而,到现在居然连第一个例子都还没有跑起来(心痛的仿佛无法呼吸)。现在我就开始讲一讲我在Hibernate第一战中就被锤了一榔头的经过。
被锤的经过是这样的:作为惯例,在学习一个新东西之前,要做一个最基本的例子。于是,我就照着书上给的案例,建立项目,导入库,写各种文件,然后运行。。。。
What the fvck!
于是,我开始在网上找这个异常的解决办法。然后就找到了以下几个会导致这个异常的原因:
- import的包不对:在使用注解来进行配置时,@Entity等一系列注解都是存在于javax.persistence包中的,而eclipse的自动补充导入包的时候,默认会导入org.hibernate.annotations包,看起来第二个更像是正确的,但是这里的注解用的都是JPA的标准注解,需要的包是第一个;
- 配置文件中映射的缺失:这个问题我在一开始的时候确实发生了,就是在hibernate.cfg.xml配置文件中,缺少了<mapping class="..." />元素,这个元素表示的是需要映射实体的类。于是,我飞快的加了上去。
- 缺少某些库:这个原因也是发生异常的源头之一。我也对照着书上和网上给出的列表,仔细地添加了进去。
最终的项目文件结构和代码如下所示:
hibernater.cfg.xml
<?xml version="1.0" encoding="GBK"?>
<!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.temp.use_jdbc_metadata_defaults">false</property>
<!-- 指定连接数据库所用的驱动 -->
<property name="connection.driver_class">com.mysql.jdbc.Driver</property>
<!-- 指定连接数据库的url,其中hibernate是本应用连接的数据库名 -->
<property name="connection.url">jdbc:mysql://localhost/hibernate</property>
<!-- 指定连接数据库的用户名 -->
<property name="connection.username">root</property>
<!-- 指定连接数据库的密码 -->
<property name="connection.password">dongzhong1990</property>
<!-- 指定连接池里最大连接数 -->
<property name="hibernate.c3p0.max_size">20</property>
<!-- 指定连接池里最小连接数 -->
<property name="hibernate.c3p0.min_size">1</property>
<!-- 指定连接池里连接的超时时长 -->
<property name="hibernate.c3p0.timeout">5000</property>
<!-- 指定连接池里最大缓存多少个Statement对象 -->
<property name="hibernate.c3p0.max_statements">100</property>
<property name="hibernate.c3p0.idle_test_period">3000</property>
<property name="hibernate.c3p0.acquire_increment">2</property>
<property name="hibernate.c3p0.validate">true</property>
<!-- 指定数据库方言 -->
<property name="dialect">org.hibernate.dialect.MySQL5InnoDBDialect</property>
<!-- 根据需要自动创建数据表 -->
<property name="hbm2ddl.auto">update</property><!--①-->
<!-- 显示Hibernate持久化操作所生成的SQL -->
<property name="show_sql">true</property>
<!-- 将SQL脚本进行格式化后再输出 -->
<property name="hibernate.format_sql">true</property>
<!-- 罗列所有持久化类的类名 -->
<!--mapping resource="org/crazyit/app/domain/News.hbm.xml"/-->
<mapping class="org.crazyit.app.domain.News"/>
</session-factory>
</hibernate-configuration>
News.java:
package org.crazyit.app.domain;
import java.io.Serializable;
import javax.persistence.*;
@Entity
@Table(name = "news_inf")
public class News implements Serializable
{
// 消息类的标识属性
private Integer id;
// 消息标题
private String title;
// 消息内容
private String content;
// id的setter和getter方法
public void setId(Integer id)
{
this.id = id;
}
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
public Integer getId()
{
return this.id;
}
// title的setter和getter方法
public void setTitle(String title)
{
this.title = title;
}
public String getTitle()
{
return this.title;
}
// content的setter和getter方法
public void setContent(String content)
{
this.content = content;
}
public String getContent()
{
return this.content;
}
}
NewsManager.java:
package lee;
import org.hibernate.*;
import org.hibernate.cfg.*;
import org.hibernate.service.*;
import org.hibernate.boot.registry.*;
import org.crazyit.app.domain.*;
public class NewsManager
{
public static void main(String[] args)
throws Exception
{
// 实例化Configuration,
// 不带参数的configure()方法默认加载hibernate.cfg.xml文件,
// 如果传入abc.xml作为参数,则不再加载hibernate.cfg.xml,改为加载abc.xml
Configuration conf = new Configuration().configure();
ServiceRegistry serviceRegistry = new StandardServiceRegistryBuilder()
.applySettings(conf.getProperties()).build();
// 以Configuration实例创建SessionFactory实例
SessionFactory sf = conf.buildSessionFactory(serviceRegistry);
// 创建Session
Session sess = sf.openSession();
// 开始事务
Transaction tx = sess.beginTransaction();
// 创建消息对象
News n = new News();
// 设置消息标题和消息内容
n.setTitle("aaaa");
n.setContent("bbbb");
// 保存消息
sess.save(n);
// 提交事务
tx.commit();
// 关闭Session
sess.close();
sf.close();
}
}
然而,还是Unknown Entity异常,炸了炸了。。。。已经试了一个星期了,然并卵。。
希望各位大神们拯救一下我这个陷入眩晕状态下的菜鸟吧。。。。。。
-------------------------------------
(2016年6月7日更新)
终于能够运行起来了,我的解决办法是在建立SessionFactory时,不加ServiceRegistry参数,如下:
ServiceRegistry serviceRegistry = new StandardServiceRegistryBuilder()
.applySettings(conf.getProperties()).build();
SessionFactory sf = conf.buildSessionFactory(serviceRegistry);
Session sess = sf.openSession();
改为:
//ServiceRegistry serviceRegistry = new StandardServiceRegistryBuilder()
.applySettings(conf.getProperties()).build();
SessionFactory sf = conf.buildSessionFactory();//这里不使用参数
Session sess = sf.openSession();
这个更改后,程序就能够正常运行了。具体的原因我还在查找。