posts - 495,  comments - 11,  trackbacks - 0

          ??????在一般情況下,在新增領(lǐng)域?qū)ο蠛螅夹枰@取對(duì)應(yīng)的主鍵值。使用應(yīng)用層來(lái)維護(hù)主鍵,在一定程度上有利于程序性能的優(yōu)化和應(yīng)用移植性的提高。在采用數(shù)據(jù)庫(kù)自增主鍵的方案里,如果JDBC驅(qū)動(dòng)不能綁定新增記錄對(duì)應(yīng)的主鍵,就需要手工執(zhí)行查詢(xún)語(yǔ)句以獲取對(duì)應(yīng)的主鍵值,對(duì)于高并發(fā)的系統(tǒng),這很容易返回錯(cuò)誤的主鍵。通過(guò)帶緩存的DataFieldMaxValueIncrementer,可以一次獲取批量的主鍵值,供多次插入領(lǐng)域?qū)ο髸r(shí)使用,它的執(zhí)行性能是很高的。

          ??????我們經(jīng)常使用數(shù)據(jù)的自增字段作為表主鍵,也即主鍵值不在應(yīng)用層產(chǎn)生,而是在新增記錄時(shí),由數(shù)據(jù)庫(kù)產(chǎn)生。這樣,應(yīng)用層在保存對(duì)象前并不知道對(duì)象主鍵值,而必須在保存數(shù)據(jù)后才能從數(shù)據(jù)庫(kù)中返回主鍵值。在很多情況下,我們需要獲取新對(duì)象持久化后的主鍵值。在Hibernate等ORM框架,新對(duì)象持久化后,Hibernate會(huì)自動(dòng)將主鍵值綁定到對(duì)象上,給程序的開(kāi)發(fā)帶來(lái)了很多方便。?

          ??????在JDBC 3.0規(guī)范中,當(dāng)新增記錄時(shí),允許將數(shù)據(jù)庫(kù)自動(dòng)產(chǎn)生的主鍵值綁定到Statement或PreparedStatement中。

          ??????使用Statement時(shí),可以通過(guò)以下方法綁定主鍵值:?int executeUpdate(String sql, int autoGeneratedKeys)?

          ??????也可以通過(guò)Connection創(chuàng)建綁定自增值的PreparedStatement:?PreparedStatement prepareStatement(String sql, int autoGeneratedKeys)?

          ??????當(dāng)autoGeneratedKeys參數(shù)設(shè)置為Statement.RETURN_GENERATED_KEYS值時(shí)即可綁定數(shù)據(jù)庫(kù)產(chǎn)生的主鍵值,設(shè)置為Statement.NO_GENERATED_KEYS時(shí),不綁定主鍵值。下面的代碼演示了Statement綁定并獲取數(shù)據(jù)庫(kù)產(chǎn)生的主鍵值的過(guò)程:

          ????? Statement stmt = conn.createStatement();
          ????? String sql = "INSERT INTO t_topic(topic_title,user_id) VALUES(‘測(cè)試主題’,’123’)";
          ????? stmt.executeUpdate(sql,Statement.RETURN_GENERATED_KEYS); // ①指定綁定表自增主鍵值
          ????? ResultSet rs = stmt.getGeneratedKeys();
          ????? if( rs.next() ) {
          ?????????? intkey = rs.getInt(); // ②獲取對(duì)應(yīng)的表自增主鍵值
          ????? }

          ??????Spring利用這一技術(shù),提供了一個(gè)可以返回新增記錄對(duì)應(yīng)主鍵值的方法:?int update(PreparedStatementCreator psc, KeyHolder generatedKeyHolder)?,其中第二個(gè)參數(shù)類(lèi)型org.springframework.jdbc.support.KeyHolder,它是一個(gè)回調(diào)接口,Spring使用它保存新增記錄對(duì)應(yīng)的主鍵,該接口的接口方法描述如下:?

          ??????Number getKey() throws InvalidDataAccessApiUsageException;

          ??????當(dāng)僅插入一行數(shù)據(jù),主鍵不是復(fù)合鍵且是數(shù)字類(lèi)型時(shí),通過(guò)該方法可以直接返回新的主鍵值。如果是復(fù)合主鍵,或者有多個(gè)主鍵返回時(shí),該方法拋出 InvalidDataAccessApiUsageException。該方法是最常用的方法,因?yàn)橐话闱闆r下,我們一次僅插入一條數(shù)據(jù)并且主鍵字段類(lèi)型為數(shù)字類(lèi)型;?

          ??????如果是復(fù)合主鍵,則列名和列值構(gòu)成Map中的一個(gè)Entry。如果返回的是多個(gè)主鍵,則拋出InvalidDataAccessApiUsageException異常;?

          ??????Map getKeys() throws InvalidDataAccessApiUsageException;

          ??????如果返回多個(gè)主鍵,即PreparedStatement新增了多條記錄,則每一個(gè)主鍵對(duì)應(yīng)一個(gè)Map,多個(gè)Map構(gòu)成一個(gè)List。

          ??????List getKeyList():?

          ??????Spring為KeyHolder接口指代了一個(gè)通用的實(shí)現(xiàn)類(lèi)GeneratedKeyHolder,該類(lèi)返回新增記錄時(shí)的自增長(zhǎng)主鍵值。假設(shè)我們希望在新增論壇板塊對(duì)象后,希望將主鍵值加載到對(duì)象中,則可以按以下代碼進(jìn)行調(diào)整:

          ????? public voidaddForum(final Forum forum) {
          ??????????? final String sql = "INSERT INTO t_forum(forum_name,forum_desc) VALUES(?,?)";
          ??????????? KeyHolder keyHolder = newGeneratedKeyHolder(); // ①創(chuàng)建一個(gè)主鍵執(zhí)有者
          ??????????? getJdbcTemplate().update(newPreparedStatementCreator() {
          ????????????????? public PreparedStatement createPreparedStatement(Connection conn) throws SQLException {
          ??????????????????????? PreparedStatement ps = conn.prepareStatement(sql);
          ??????????????????????? ps.setString(1, forum.getForumName());
          ??????????????????????? ps.setString(2, forum.getForumDesc());
          ??????????????????????? returnps;
          ????????????????? }
          ??????????? }, keyHolder);
          ??????????? forum.setForumId(keyHolder.getKey().intValue()); // ②從主鍵執(zhí)有者中獲取主鍵
          ????? }

          ??????這樣,在調(diào)用addForum(Forum forum)新增forum領(lǐng)域?qū)ο蠛螅琭orum將擁有對(duì)應(yīng)的主鍵值,方便后繼的使用。在JDBC 3.0之前的版本中,PreparedStatement不能綁定主鍵,如果采用表自增鍵(如MySQL的auto increment或SQLServer的identity)將給獲取正確的主鍵值帶來(lái)挑戰(zhàn)——因?yàn)槟惚仨氃诓迦霐?shù)據(jù)后,馬上執(zhí)行另一條獲取新增主鍵的查詢(xún)語(yǔ)句。下面給出了不同數(shù)據(jù)庫(kù)獲取最新自增主鍵值的查詢(xún)語(yǔ)句:?

          posted on 2011-09-25 14:27 jadmin 閱讀(998) 評(píng)論(0)  編輯  收藏

          只有注冊(cè)用戶(hù)登錄后才能發(fā)表評(píng)論。


          網(wǎng)站導(dǎo)航:
           
          主站蜘蛛池模板: 永嘉县| 桑日县| 东城区| 镇雄县| 华蓥市| 焦作市| 特克斯县| 五峰| 高阳县| 田东县| 成都市| 镇安县| 泗水县| 安丘市| 教育| 河东区| 盐池县| 青冈县| 洪洞县| 五峰| 平泉县| 广河县| 达日县| 桂东县| 灵宝市| 成安县| 庆元县| 吐鲁番市| 兴安盟| 民勤县| 利川市| 桦南县| 邵阳县| 威海市| 酒泉市| 逊克县| 延津县| 涪陵区| 日土县| 托克逊县| 如东县|