我和hibernate的第一次親密:
我用的數據庫是mssql,用middlegen稀里嘩啦生成一堆文件。編譯運行,hibernate報告一堆異常,大概是說,MSSQL JDBC抗議說他不能讀取已經讀過的字段,所以hibernate不能生成需要的collection。最后icepeak傳給我一個JSQLConnect,終于把這個問題解決了。
設置hibernate生成定制的主鍵ID:
在和hibernate親密接觸之后,我想按照自己的想法來設計組織機構.對于Organ,Department,Post和Stuff.我希望這樣設計主鍵的ID,類型都是char,格式為:
Organ0000000001
Dept0000000001
Post0000000001
Stuff0000000002
修改middlegen產生的.hbm.xml文件:
......
<class
??? name="com.abcdefg.hibernate.Organ"
??? table="Organ"
>
??? <id
??????? name="organId"
??????? type="java.lang.String"
??????? column="OrganId"
??? >
??????? <generator class="hbtest.PkGenerator">
??????????? <param name="table">PKGenerator_table</param>
??????????? <param name="column">NextOrganId</param>
??????????? <param name="max_lo">0</param>
?</generator>
??? </id>
......
</class>
找到的資料說:"如果需要采用定制的主鍵產生算法,則在此處配置主鍵生成器,主鍵生成器必須實現net.sf.hibernate.id.IdentifierGenerator 接口".我找了半天都沒找到net.sf.hibernate.id.IdentifierGenerator.其實只要從hibernate自身提供的一些主鍵生成器繼承就可以,從hibernate的源文件可以看到這些主鍵生成器實現的接口是org.hibernate.id
我的PkGenerator.java文件:
package hbtest;
import org.hibernate.id.TableHiLoGenerator;
import java.io.Serializable;
import java.util.Properties;
import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.hibernate.HibernateException;
import org.hibernate.dialect.Dialect;
import org.hibernate.engine.SessionImplementor;
import org.hibernate.type.Type;
import org.hibernate.type.IntegerType;
import org.hibernate.util.PropertiesHelper;
public class PkGenerator
??? extends TableHiLoGenerator {
? private String columnName;
? public void configure(Type type, Properties params, Dialect d) {
??? super.configure(new IntegerType(), params, d);
??? this.columnName = PropertiesHelper.getString(COLUMN, params,
???????????????????????????????????????????????? DEFAULT_COLUMN_NAME);
? }
? public synchronized Serializable generate(SessionImplementor session,
??????????????????????????????????????????? Object obj) throws
????? HibernateException {
??? Object result = super.generate(session, obj);
??? String strNumber = result.toString();
??? StringBuffer strResult = new StringBuffer("");
??? if (columnName.compareToIgnoreCase("NextOrganId") == 0) {
????? strResult.append("Organ");
??? }
??? else if(columnName.compareToIgnoreCase("NextDeptId") == 0){
????? strResult.append("Dept");
??? }
??? ......
??? if (strNumber.length() > 10)
????? throw new HibernateException("The generated " + strResult +
?????????????????????????????????? "Id is too long!");
??? else if (strNumber.length() == 10) {
????? strResult.append(strNumber);
??? }
??? else {
??????? int count = 10 - strNumber.length();
??????? while (count > 0) {
??????? strResult.append("0");
??????? count--;
????? }
????? strResult.append(strNumber);
??? }
??? return strResult.toString();
? }
}
實現了以后發現在這些主鍵前面添加"Organ","Dept"之類的并不是特別有用,而且我開始的出發點還是錯的.也許我應該建立更多的映射表...?
Hibernate的初級指南
hibernate貼心的地方:
驗證了一下,我的擔心終于被證實是多余的了,我擔心的是hibernate會為同一條記錄產生一堆不同的對象。
我覺得使hibernate真正實用的條件:為同一條記錄產生的多個對象是同一個對象。當然前提是設置了主鍵。
hibernate也許是將當前讀出的對象保存在一個哈希表,在讀一個新對象的時候會先從這個哈希表中查找。好像這并不是多困難的事情。
?
但多線程情況下就有所不同了..
因為Hibernate說(對于Session):
It is not intended that implementors be threadsafe. Instead each thread/transaction should obtain its own instance from a SessionFactory
?
Session并沒有要求是線程安全的,而通過實驗表明記錄和對象的一一對應只存在于同一個Session中。這也是hibernate的文檔中HibernateUtil類使用ThreadLocal的原因吧。
所以對于多線程的情況,在不同的線程中,對應同一條記錄的對象是不同的。
(另hibernate有二級緩存,是全局的緩存,可以用來減少數據庫負載,二級緩存的功能需要設置才會啟用)
Hibernate中修改多對多關系的屬性:<middlegen>項中有如下<table>配置:
?? <table generate="true" name="Post"/>
?? <table generate="true" name="Stuff"/>
?? <many2many>
????? <tablea generate="true" name="Post" />
????? <jointable name="PostStuff" generate="true" />
????? <tableb generate="true" name="Stuff" />
?? </many2many>
即表Post與Stuff的關系是多對多,而且表PostStuff中一個字段時JoinDate:員工加入部門的時間。
我現在想加入一條Post記錄,一條Stuff記錄,然后設置Stuff加入Post的時間。經過多次嘗試,代碼如下:
?? ...
?? Dept dept = ....
?? Post post = new Post("third post1",dept,new HashSet(),new HashSet());
?? Stuff stuff = new Stuff("third stuff",new HashSet(),new HashSet());
?? session.save(stuff);
?? session.save(post);
?? post.getStuffs().add(stuff);
?? //stuff.getPosts().add(post);/*不能與上一行并存,會有異常,因為是一個死循環?*/
?? PostStuff postStuff = (PostStuff)session.createQuery("from PostStuff where PostId = '" + post.getPostId() + "' AND StuffId = '" + stuff.getStuffId() + "'").iterate().next();
?? postStuff.setJoinDate(new java.util.Date());
?? tx.commit();
?? HibernateUtil.closeSession();
?? ...
post.getStuffs().add(stuff);在兩個Save()之后,而且沒有調用session.flush(),但是createQuery能夠得到PostStuff說明HQL是先從內存中查詢數據的。不需要調用session.flush()來寫入數據庫,因為tx.commit()會自動調用。
可以用session.persist(stuff)來替換session.save(stuff),我暈,明明save有說明會創建id,而persist沒有說明會創建id的(但實際上是創建了)。
試圖用以下替代post.getStuffs().add(stuff):
?? PostStuff postStuff = new PostStuff();
?? postStuff.setPost(post);
?? postStuff.setStuff(stuff);
?? postStuff.setJoinDate(new java.util.Date());
?? postStuff.setComp_id(new PostStuffPK(post.getPostId(),stuff.getStuffId()));
?? post.getPostStuffs().add(postStuff);
結果是雖然沒有異常,但并不會添加一條關系。
或者
?? session.flush();
?? session.refresh(stuff);
?? ((PostStuff)(stuff.getPostStuffs().iterator().next())).setJoinDate(new java.util.Date());
注意這些代碼只是用來作試驗用的