📄 quickstart.html
字号:
<property name="weight"/> </class></hibernate-mapping></pre><p> 每个持久化类都需要一个标识属性(实际上,只是哪些代表一手对象的类,而不是代表值对象的类,后者会被映射称为一手对象中的一个组件)。这个属性用来区分持久化对象:如果<tt class="literal">catA.getId().equals(catB.getId())</tt>结果是true的话,两只猫就是相同的。这个概念称为<span class="emphasis"><em>数据库标识</em></span>。Hiernate附带了几种不同的标识符生成器,用于不同的场合(包括数据库本地的顺序(sequence)生成器和hi/lo高低位标识模式)。我们在这里使用UUID生成器,并指定<tt class="literal">CAT</tt>表的<tt class="literal">CAT_ID</tt>字段(作为表的主键)存放生成的标识值。 </p><p> <tt class="literal">Cat</tt>的其他属性都映射到同一个表。对<tt class="literal">name</tt>属性来说,我们把它显式地声明映射到一个数据库字段。如果数据库schema是由映射声明使用Hibernate的<span class="emphasis"><em>SchemaExport</em></span>工具自动生成的(作为SQL DDL指令),这特别有用。所有其它的属性都用Hibernate的默认值映射,大多数情况你都会这样做。数据库中的<tt class="literal">CAT</tt>表看起来是这样的: </p><pre class="programlisting"> Column | Type | Modifiers--------+-----------------------+----------- cat_id | character(32) | not null name | character varying(16) | not null sex | character(1) | weight | real |Indexes: cat_pkey primary key btree (cat_id)</pre></div><p> 你现在可以在你的数据库中首先创建这个表了,如果你需要使用SchemaExport工具把这个步骤自动化,请参阅<a href="toolsetguide.html" title="Chapter 19. 工具箱指南">Chapter 19, <i>工具箱指南</i></a>。这个工具能够创建完整的SQL DDL,包括表定义,自定义的字段类型约束,惟一约束和索引。 </p><div class="sect1" lang="en"><div class="titlepage"><div><div><h2 class="title" style="clear: both"><a name="d0e318"></a>1.4. 与猫同乐</h2></div></div><div></div></div><p> 我们现在可以开始Hibernate的<tt class="literal">Session</tt>了。我们用它来从数据库中存取<tt class="literal">Cat</tt>。首先,我们要从<tt class="literal">SessionFactory</tt>中获取一个<tt class="literal">Session</tt>(Hibernate的工作单元)。 </p><pre class="programlisting">SessionFactory sessionFactory = new Configuration().configure().buildSessionFactory();</pre><p> <tt class="literal">SessionFactory</tt>负责一个数据库,也只对应一个XML配置文件(<tt class="literal">hibernate.cfg.xml</tt>)。 </p><p> 这篇教程的关注点在于配置Tomcat的JDBC连接,绑定到JNDI上,以及Hibernate的基础配置。你可以用喜欢的任何方式编写一个Servlet,包含下面的代码,只要确保<tt class="literal">SessionFactory</tt>只创建一次。也就是说你不能把它作为你的Serlvet 的实例变量。一个好办法是用在辅助类中用一个静态的<tt class="literal">SessionFactory</tt>,例如这样: </p><pre class="programlisting">import net.sf.hibernate.*;import net.sf.hibernate.cfg.*;public class HibernateUtil { private static final SessionFactory sessionFactory; static { try { sessionFactory = new Configuration().configure().buildSessionFactory(); } catch (HibernateException ex) { throw new RuntimeException("Exception building SessionFactory: " + ex.getMessage(), ex); } } public static final ThreadLocal session = new ThreadLocal(); public static Session currentSession() throws HibernateException { Session s = (Session) session.get(); // Open a new Session, if this Thread has none yet if (s == null) { s = sessionFactory.openSession(); session.set(s); } return s; } public static void closeSession() throws HibernateException { Session s = (Session) session.get(); session.set(null); if (s != null) s.close(); }}</pre><p> 这个类不但在它的静态属性中使用了<tt class="literal">SessionFactory</tt>,还使用了<tt class="literal">ThreadLocal</tt>来为当前工作线程保存<tt class="literal">Session</tt>。 </p><p> <tt class="literal">Session</tt>不是线程安全的,代表与数据库之间的一次操作。<tt class="literal">Session</tt>通过<tt class="literal">SessionFactory</tt>打开,在所有的工作完成后,需要关闭: </p><pre class="programlisting">Session session = HibernateUtil.currentSession();Transaction tx= session.beginTransaction();Cat princess = new Cat();princess.setName("Princess");princess.setSex('F');princess.setWeight(7.4f);session.save(princess);tx.commit();HibernateUtil.closeSession();</pre><p> 在<tt class="literal">Session</tt>中,每个数据库操作都是在一个事务(transaction)中进行的,这样就可以隔离开不同的操作(甚至包括只读操作)。我们使用Hibernate的<tt class="literal">Transaction</tt> API来从底层的事务策略中(本例中是JDBC事务)脱身。这样,如果需要把我们的程序部署到一个由容器管理事务的环境中去(使用JTA),我们就不需要更改源代码。请注意,我们上面的例子没有处理任何异常。 </p><p> 也请注意,你可以随心所欲的多次调用<tt class="literal">HibernateUtil.currentSession();</tt>,你每次都会得到同一个当前线程的<tt class="literal">Session</tt>。你必须确保<tt class="literal">Session</tt>在你的数据库事务完成后关闭,不管是在你的Servlet代码中,或者在ServletFilter中,HTTP结果返回之前。 </p><p> Hibernate有不同的方法来从数据库中取回对象。最灵活的方式是使用Hibernate查询语言(HQL),这是一种容易学习的语言,是对SQL的面向对象的强大扩展。 </p><pre class="programlisting">Transaction tx= session.beginTransaction();Query query = session.createQuery("select cat from Cat as cat where cat.sex = :sex");query.setCharacter("sex", 'F');for (Iterator it = query.iterate(); it.hasNext();) { Cat cat = (Cat) it.next(); out.println("Female Cat: " + cat.getName() );}tx.commit();</pre><p> Hibernate也提供一种面向对象的<span class="emphasis"><em>按条件查询</em></span>API,可以执行公式化的类型安全的查询。当然,Hibernate在所有与数据库的交互中都使用<tt class="literal">PrepatedStatement</tt>和参数绑定。 </p></div><div class="sect1" lang="en"><div class="titlepage"><div><div><h2 class="title" style="clear: both"><a name="d0e410"></a>1.5. 结语</h2></div></div><div></div></div><p> 在这个短小的教程中,我们只描绘了Hibernate的基本面目。请注意我们没有在例子中包含Servlet相关代码。你必须自行编写Servlet,然后把你认为合适的Hibernate代码插入。 </p><p> 请记住Hibernate作为数据库访问层,是与你的程序紧密相关的。一般,所有其他层次都依赖持久机制。请确信你理解了这种设计的含义。 </p></div></div><div class="navfooter"><hr><table width="100%" summary="Navigation footer"><tr><td width="40%" align="left"><a accesskey="p" href="pr01.html">Prev</a> </td><td width="20%" align="center"><a accesskey="u" href="index.html">Up</a></td><td width="40%" align="right"> <a accesskey="n" href="architecture.html">Next</a></td></tr><tr><td width="40%" align="left" valign="top">前言 </td><td width="20%" align="center"><a accesskey="h" href="index.html">Home</a></td><td width="40%" align="right" valign="top"> Chapter 2. 体系结构</td></tr></table></div></body></html>
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -