• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            brent's hut

            關(guān)于Hibernate的記錄

            我和hibernate的第一次親密:

            我用的數(shù)據(jù)庫(kù)是mssql,用middlegen稀里嘩啦生成一堆文件。編譯運(yùn)行,hibernate報(bào)告一堆異常,大概是說,MSSQL JDBC抗議說他不能讀取已經(jīng)讀過的字段,所以hibernate不能生成需要的collection。最后icepeak傳給我一個(gè)JSQLConnect,終于把這個(gè)問題解決了。

            設(shè)置hibernate生成定制的主鍵ID

            在和hibernate親密接觸之后,我想按照自己的想法來設(shè)計(jì)組織機(jī)構(gòu).對(duì)于Organ,Department,Post和Stuff.我希望這樣設(shè)計(jì)主鍵的ID,類型都是char,格式為:
            Organ0000000001
            Dept0000000001
            Post0000000001
            Stuff0000000002

            修改middlegen產(chǎn)生的.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>

            找到的資料說:"如果需要采用定制的主鍵產(chǎn)生算法,則在此處配置主鍵生成器,主鍵生成器必須實(shí)現(xiàn)net.sf.hibernate.id.IdentifierGenerator 接口".我找了半天都沒找到net.sf.hibernate.id.IdentifierGenerator.其實(shí)只要從hibernate自身提供的一些主鍵生成器繼承就可以,從hibernate的源文件可以看到這些主鍵生成器實(shí)現(xiàn)的接口是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();
            ? }
            }

            實(shí)現(xiàn)了以后發(fā)現(xiàn)在這些主鍵前面添加"Organ","Dept"之類的并不是特別有用,而且我開始的出發(fā)點(diǎn)還是錯(cuò)的.也許我應(yīng)該建立更多的映射表...?

            Hibernate的初級(jí)指南

            hibernate貼心的地方

            驗(yàn)證了一下,我的擔(dān)心終于被證實(shí)是多余的了,我擔(dān)心的是hibernate會(huì)為同一條記錄產(chǎn)生一堆不同的對(duì)象。
            我覺得使hibernate真正實(shí)用的條件:為同一條記錄產(chǎn)生的多個(gè)對(duì)象是同一個(gè)對(duì)象。當(dāng)然前提是設(shè)置了主鍵。
            hibernate也許是將當(dāng)前讀出的對(duì)象保存在一個(gè)哈希表,在讀一個(gè)新對(duì)象的時(shí)候會(huì)先從這個(gè)哈希表中查找。好像這并不是多困難的事情。
            ?
            但多線程情況下就有所不同了..
            因?yàn)镠ibernate說(對(duì)于Session):
            It is not intended that implementors be threadsafe. Instead each thread/transaction should obtain its own instance from a SessionFactory
            ?
            Session并沒有要求是線程安全的,而通過實(shí)驗(yàn)表明記錄和對(duì)象的一一對(duì)應(yīng)只存在于同一個(gè)Session中。這也是hibernate的文檔中HibernateUtil類使用ThreadLocal的原因吧。
            所以對(duì)于多線程的情況,在不同的線程中,對(duì)應(yīng)同一條記錄的對(duì)象是不同的。

            (另hibernate有二級(jí)緩存,是全局的緩存,可以用來減少數(shù)據(jù)庫(kù)負(fù)載,二級(jí)緩存的功能需要設(shè)置才會(huì)啟用)

            Hibernate中修改多對(duì)多關(guān)系的屬性:

            <middlegen>項(xiàng)中有如下<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的關(guān)系是多對(duì)多,而且表PostStuff中一個(gè)字段時(shí)JoinDate:員工加入部門的時(shí)間。

            我現(xiàn)在想加入一條Post記錄,一條Stuff記錄,然后設(shè)置Stuff加入Post的時(shí)間。經(jīng)過多次嘗試,代碼如下:
            ?? ...
            ?? 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);/*不能與上一行并存,會(huì)有異常,因?yàn)槭且粋€(gè)死循環(huán)?*/
            ?? 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);在兩個(gè)Save()之后,而且沒有調(diào)用session.flush(),但是createQuery能夠得到PostStuff說明HQL是先從內(nèi)存中查詢數(shù)據(jù)的。不需要調(diào)用session.flush()來寫入數(shù)據(jù)庫(kù),因?yàn)閠x.commit()會(huì)自動(dòng)調(diào)用。

            可以用session.persist(stuff)來替換session.save(stuff),我暈,明明save有說明會(huì)創(chuàng)建id,而persist沒有說明會(huì)創(chuàng)建id的(但實(shí)際上是創(chuàng)建了)。

            試圖用以下替代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);
            結(jié)果是雖然沒有異常,但并不會(huì)添加一條關(guān)系。

            或者

            ?? session.flush();
            ?? session.refresh(stuff);
            ?? ((PostStuff)(stuff.getPostStuffs().iterator().next())).setJoinDate(new java.util.Date());
            注意這些代碼只是用來作試驗(yàn)用的

            posted on 2005-12-19 09:23 brent 閱讀(534) 評(píng)論(0)  編輯 收藏 引用 所屬分類: Java

            77777亚洲午夜久久多喷| 国产99久久九九精品无码| 99久久国产综合精品女同图片 | 久久亚洲AV成人无码软件| 国产美女亚洲精品久久久综合| 精品永久久福利一区二区| 久久99国产一区二区三区| 囯产极品美女高潮无套久久久 | 亚洲国产二区三区久久| 偷偷做久久久久网站| 国产精品99久久久久久董美香| 精品一二三区久久aaa片| 久久精品国产精品亚洲下载| 精品久久久久久久无码 | 四虎国产精品成人免费久久| 久久久久久亚洲AV无码专区| 理论片午午伦夜理片久久| 国产精品久久久久无码av| 日韩精品久久无码人妻中文字幕| 久久无码AV中文出轨人妻 | 久久久久久久综合狠狠综合| 国产成人综合久久久久久| 久久国产精品99久久久久久老狼 | 久久综合狠狠综合久久激情 | 香蕉久久夜色精品升级完成| 狠狠色丁香婷婷久久综合| 精品久久久久国产免费| 国产免费福利体检区久久| 久久精品水蜜桃av综合天堂| 久久亚洲AV成人出白浆无码国产| 久久国产亚洲精品| 久久这里只精品99re66| 久久婷婷色综合一区二区| 久久婷婷五月综合国产尤物app| 精品久久久久成人码免费动漫 | 色偷偷偷久久伊人大杳蕉| 亚洲精品乱码久久久久久久久久久久 | 久久一区二区三区免费| 亚洲第一永久AV网站久久精品男人的天堂AV| 国产成人精品综合久久久| 国产真实乱对白精彩久久|