亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 數據庫 > Oracle > 正文

在Spring中用select last_insert_id()時遇到問題

2020-07-26 14:35:11
字體:
來源:轉載
供稿:網友
今天在使用MySQL時卻不知如何處理,插入記錄后不知怎樣獲得剛剛插入的id,查過文檔后發現了select last_insert_id(),在插入之后執行此查詢,即可獲得自增id,喜出望外。
可用到自己的程序中之后卻得不到想要的結果,于是就懷疑到了Spring頭上,因為通過基本JDBC測試是沒有任何問題的,所以就去跟蹤Spring JDBC, 看過源碼之后才豁然開朗,原來Spring中如此獲得數據庫Connection的:Connection con DataSourceUtils.getConnection(getDataSource());, 哎,只能怪自己Spring掌握不夠好,所以就不能在執行insert之后去執行select last_insert_id()了,因為select last_insert_id()是真對當前Connection插入和更新操作的,那在Spring中如何獲得新增記錄的鍵值呢,沒有辦法只能去Google了。

發現了如下文章,來自IT168,標題為《Spring應用數據主鍵的生成策略盤點》,摘錄如下:


    在一般情況下,在新增領域對象后,都需要獲取對應的主鍵值。使用應用層來維護主鍵,在一定程度上有利于程序性能的優化和應用移植性的提高。在采用數據庫自增主鍵的方案里,如果JDBC驅動不能綁定新增記錄對應的主鍵,就需要手工執行查詢語句以獲取對應的主鍵值,對于高并發的系統,這很容易返回錯誤的主鍵。通過帶緩存的DataFieldMaxValueIncrementer,可以一次獲取批量的主鍵值,供多次插入領域對象時使用,它的執行性能是很高的。

使用數據庫的自增主鍵
    我們經常使用數據的自增字段作為表主鍵,也即主鍵值不在應用層產生,而是在新增記錄時,由數據庫產生。這樣,應用層在保存對象前并不知道對象主鍵值,而必須在保存數據后才能從數據庫中返回主鍵值。在很多情況下,我們需要獲取新對象持久化后的主鍵值。在Hibernate等ORM框架,新對象持久化后,Hibernate會自動將主鍵值綁定到對象上,給程序的開發帶來了很多方便。

    在JDBC 3.0規范中,當新增記錄時,允許將數據庫自動產生的主鍵值綁定到Statement或PreparedStatement中。使用Statement時,可以通過以下方法綁定主鍵值:
int executeUpdate(String sql,int autoGeneratedKeys)
也可以通過Connection創建綁定自增值的PreparedStatement:
PreparedStatement prepareStatement(String sql, int autoGeneratedKeys)

    當autoGeneratedKeys參數設置為Statement.RETURN_GENERATED_KEYS值時即可綁定數據庫產生的主鍵值,設置為Statement.NO_GENERATED_KEYS時,不綁定主鍵值。下面的代碼演示了Statement綁定并獲取數據庫產生的主鍵值的過程:
Statement stmt = conn.createStatement(); String sql = "INSERT INTO t_topic(topic_title,user_id) VALUES(‘測試主題','123') "; stmt.executeUpdate(sql,Statement.RETURN_GENERATED_KEYS); ①指定綁定表自增主鍵值 ResultSet rs = stmt.getGeneratedKeys(); if ( rs.next() ) { int key = rs.getInt();②獲取對應的表自增主鍵值 }



Spring利用這一技術,提供了一個可以返回新增記錄對應主鍵值的方法:
int update(PreparedStatementCreator psc, KeyHolder generatedKeyHolder)
org.springframework.jdbc.support.KeyHolder是一個回調接口,Spring使用它保存新增記錄對應的主鍵,該接口的接口方法描述如下:
 Number getKey() throws InvalidDataAccessApiUsageException
當 僅插入一行數據,主鍵不是復合鍵且是數字類型時,通過該方法可以直接返回新的主鍵值。如果是復合主鍵,或者有多個主鍵返回時,該方法拋出 InvalidDataAccessApiUsageException。該方法是最常用的方法,因為一般情況下,我們一次僅插入一條數據并且主鍵字段類型為數字類型;

 Map getKeys() throws InvalidDataAccessApiUsageException
如果是復合主鍵,則列名和列值構成Map中的一個Entry。如果返回的是多個主鍵,則該方法拋出InvalidDataAccessApiUsageException異常;

 List getKeyList():
如果返回多個主鍵,即PreparedStatement新增了多條記錄,則每一個主鍵對應一個Map,多個Map構成一個List。

    Spring為KeyHolder接口指代了一個通用的實現類GeneratedKeyHolder,該類返回新增記錄時的自增長主鍵值。假設我們希望在新增論壇板塊對象后,希望將主鍵值加載到對象中,則可以按以下代碼進行調整:

 
public void addForum(final Forum forum) { final String sql = "INSERT INTO t_forum(forum_name,forum_desc) VALUES(?,?)"; KeyHolder keyHolder = new GeneratedKeyHolder();①創建一個主鍵執有者 getJdbcTemplate().update(new PreparedStatementCreator(){ public PreparedStatement createPreparedStatement(Connection conn) throws SQLException { PreparedStatement ps = conn.prepareStatement(sql); ps.setString(1, forum.getForumName()); ps.setString(2, forum.getForumDesc()); return ps; } },keyHolder); forum.setForumId(keyHolder.getKey().intValue());②從主鍵執有者中獲取主鍵 }

     這樣,在調用addForum(final Forum forum)新增forum領域對象后,forum將擁有對應的主鍵值,方便后繼的使用。
在JDBC 3.0之前的版本中,PreparedStatement不能綁定主鍵,如果采用表自增鍵(如MySql的auto increment或SqlServer的identity)將給獲取正確的主鍵值帶來挑戰――因為你必須在插入數據后,馬上執行另一條獲取新增主鍵的查詢語句。表 1給出了不同數據庫獲取最新自增主鍵值的查詢語句:
       表 1 不同數據庫獲取新增加的主鍵值 

 

數據庫
獲取新增主鍵的查詢語句
DB2
IDENTITY_VAL_LOCAL()
Informix
SELECT dbinfo('sqlca.sqlerrd1') FROM <TABLE>
Sybase
SELECT @@IDENTITY
SqlServer
SELECT SCOPE_IDENTITY()或SELECT @@IDENTITY
MySql
SELECT LAST_INSERT_ID()
HsqlDB
CALL IDENTITY()
Cloudscape
IDENTITY_VAL_LOCAL()
Derby
IDENTITY_VAL_LOCAL()
PostgreSQL
SELECT nextval('<TABLE>_SEQ')
   
    如果數據庫的并發率很高,比如在插入記錄后執行查詢主鍵之前,數據庫又執行了若干條插入記錄的SQL語句,這時,通過表 1 返回的主鍵值就是最后一條插入語句的主鍵值,而非我們希望的主鍵值了。所以使用查詢語句獲取表自增鍵值是不安全的,這也是為什么有些數據庫(如 Oracle、Firebird)故意不提供自增鍵,而只提供序列的原因,序列強制要求你在新增記錄前,先獲取主鍵值。Oracle通過SELECT <SEQUENCE_NAME>.nextval FROM DUAL獲取序列的下一個值,而FireBird通過SELECT GEN_ID(<SEQUENCE_NAME> 1) FROM RDB$DATABASE獲取序列的下一個值。在10.4.1小節中,我們還將講解應用層自增鍵的相關知識。


應用層產生主鍵
    Spring JDBC提供了自增鍵以及行集的支持,自增鍵對象讓我們可以不依賴數據庫的自增鍵,在應用層為新記錄提供主鍵值。在JDK 1.4中引入了RowSet,它允許在連接斷開的情況下操作數據,在這節里,我們將介紹如何在Spring JDBC中使用RowSet。

自增鍵的使用
    一般數據庫都提供了自增鍵的功能,如MySql的auto_increment、SqlServerr的identity字段等。Spring允許你在應用層產生主鍵值,為此定義了 org.springframework.jdbc.support.incrementer.DataFieldMaxValueIncrementer 接口,提供兩種產生主鍵的方案:第一,通過序列產生主鍵;第二,通過表產生主鍵。根據主鍵產生方式和數據庫的不同,Spring提供了若干實現類,如圖 1所示: 

      圖 1 DateFieldValueIncrementer繼承類圖
    根據不同的主鍵產生方式,可能需要配置表名、主鍵字段名或序列名等信息。下面,我們以Oracle和MySql為例分別講解使用序列及表字段產生主鍵值的方式。

DataFieldMaxValueIncrementer接口定義了3個獲取下一個主鍵值的方法:
 int nextIntValue():獲取下一個主鍵值,主鍵數據類型為int;
 long nextLongValue():獲取下一個主鍵值,主鍵數據類型為long;
 String nextStringValue():獲取下一個主鍵值,主鍵數據類型為String;
    在其抽象實現類AbstractDataFieldMaxValueIncrementer中,提供了幾個重要的屬性,其中 incrementerName定義序列或主鍵表的名稱;如果返回的主鍵是String類型,則paddingLength屬性可能會派上用場,它允許你指定返回主鍵的長度,不足的部分前面補0。

    HsqlMaxValueIncrementer和MySQLMaxValueIncrementer兩個主鍵值產生器基于表進行工作。通過 columnName屬性定義主鍵列的名字,通過cacheSize屬性定義緩存的主鍵個數,當內存中的主鍵值用完后,產生器將一次性獲取 cacheSize個主鍵,這樣可以減少數據訪問的次數,提高應用的性能。

    我們通過DateFieldValueIncrementer從數據庫中獲取主鍵值來彌補這個缺陷。首先,調整PostJdbcDao的代碼,添加DateFieldValueIncrementer屬性,并通過它從序列中得到下一個主鍵值:
    代碼清單 13 使用DateFieldValueIncrementer產生主鍵
 
public class PostJdbcDao extends JdbcDaoSupport implements PostDao { private DataFieldMaxValueIncrementer incre; ①主鍵值產生器 public void addPost(final Post post) { … getJdbcTemplate().execute( sql,new AbstractLobCreatingPreparedStatementCallback( this.lobHandler) { protected void setValues(PreparedStatement ps, LobCreator lobCreator) throws SQLException { ps.setInt(1, incre.nextIntValue());②獲取下一個主鍵值 … } }); } …//省略get/setter方法 }

在②處,我們通過incre.nextIntValue()獲取下一個主鍵值。


以序列方式產生主鍵值
    在Oracle數據庫中創建一個seq_post_id序列,使用這個序列為t_post提供主鍵值,以下是創建seq_post_id的腳本: 
 
create sequence seq_post_id increment by 1start with 1;


   接著,調整Spring的配置,使用OracleSequenceMaxValueIncrementer作為主鍵產生器:

 
<bean id="incre" class="org.springframework.jdbc.support.incrementer.OracleSequenceMaxValueIncrementer"><property name="incrementerName" value="seq_post_id"/> ①指定序列名 <property name="dataSource" ref="dataSource"/> ②設置數據源 </bean><bean id="postDao" parent="dao" class="com.baobaotao.dao.jdbc.PostJdbcDao"><property name="lobHandler" ref="oracleLobHandler"/><property name="incre" ref="incre"/> ③添加主鍵主鍵產生器 </bean>

  以表方式產生主鍵值 
 在Mysql中創建一張用于維護t_post主鍵的t_post_id表,以下是創建該表及插入初始化的SQL腳本:

 
create table t_post_id(sequence_id int) type = MYISAM;
insert into t_post_id values(0
);

    由于主鍵維護表的并發訪問量很大,所以最好將其聲明為MYISAM類型,此外需要為該表提供初始值,以便后續主鍵值在此之上進行遞增。
調整為MySql數據庫后,我們僅需要對Spring配置進行小小的調整就可以了:

 
<bean id="incre"class="org.springframework.jdbc.support.incrementer.MySQLMaxValueIncrementer"><property name="incrementerName" value="t_post_id"/> ①設置維護主鍵的表名 <property name="columnName" value="sequence_id"/>②用于生成主鍵值的列名 <property name="cacheSize" value="10"/> ③緩存大小 <property name="dataSource" ref="dataSource"/></bean><bean id="postDao" parent="dao" class="com.baobaotao.dao.jdbc.PostJdbcDao"><property name="lobHandler" ref="defaultLobHandler"/><property name="incre" ref="incre"/></bean>
    incrementerName和columnName都很容易理解,cacheSize決定一次返回的主鍵個數,這里我們設置為10。當第一次通過 MySQLMaxValueIncrementer# nextIntValue()獲取主鍵值時,MySQLMaxValueIncrementer將使t_post_id. sequence_id遞增10,而后續9次調用nextIntValue()方法時,都從緩存中獲取主鍵值。直到第10次再次調用 nextIntValue()方法時,才會再次將t_post_id. sequence_id字段值遞增10,如此循環反復。

小結
    主鍵的生產方式從產生地點上可以分為應用層產生和數據庫產生兩種方式。應用層借助數據庫的序列或表產生主鍵,這種方式可以保證程序的可移植性和安全性,同時可以通過緩存機制提高運行效率。有些數據庫支持數據表自增鍵的主鍵產生機制,在JDBC 3.0以前的版本中,無法通過Statement自動獲取新增記錄的對應主鍵。這時需要在插入數據后,馬上執行一條數據庫相關的主鍵獲取SQL語句以得到對應的主鍵值,在數據庫高并發的情況下,有可能獲取到不正確的主鍵值。在這種情況下,在插入數據前事先在應用層準備好主鍵值是很好的備選方案。


另外補充一點在SqlUpdate執行update之前需設置setReturnGeneratedKeys(true);
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
51精品国产黑色丝袜高跟鞋| 美日韩丰满少妇在线观看| 国产精品中文字幕久久久| 亚洲国产三级网| 国产福利视频一区二区| 国产精品亚洲аv天堂网| 久久全球大尺度高清视频| 国产精品入口免费视| 欧美一级电影免费在线观看| 在线电影av不卡网址| 97视频在线播放| 亚洲九九九在线观看| 国产日产久久高清欧美一区| 久久精品国产视频| 亚洲国产精品人人爽夜夜爽| 日本精品视频在线播放| 91影院在线免费观看视频| 91色视频在线观看| xvideos亚洲人网站| 亚洲风情亚aⅴ在线发布| 亚洲午夜久久久影院| 欧美又大又硬又粗bbbbb| 日韩欧中文字幕| 一区二区在线视频| 一道本无吗dⅴd在线播放一区| 欧美日韩国产成人高清视频| 中文字幕一精品亚洲无线一区| 欧美日韩亚洲天堂| 亚洲缚视频在线观看| 日韩在线高清视频| 日韩中文字幕在线视频| 丁香五六月婷婷久久激情| 欧美限制级电影在线观看| 久久69精品久久久久久国产越南| 国产日韩欧美在线观看| 久久综合久久八八| 国产亚洲欧美视频| 热re99久久精品国产66热| 91久久精品久久国产性色也91| 国产成人avxxxxx在线看| 欧美精品电影免费在线观看| 久久精品福利视频| 久久影院资源网| 懂色av影视一区二区三区| 欧美激情亚洲精品| 在线a欧美视频| 日本伊人精品一区二区三区介绍| 亚洲国产日韩欧美综合久久| 国产日产欧美a一级在线| 欧美天堂在线观看| 91亚洲精品久久久久久久久久久久| 亚洲精品videossex少妇| www国产精品视频| 中文字幕亚洲激情| 久久天天躁狠狠躁夜夜爽蜜月| 欧美一级在线亚洲天堂| 亚洲男人的天堂网站| 久久精品99无色码中文字幕| 欧美人交a欧美精品| 欧美在线亚洲一区| 国产视频福利一区| 国产精品999| 欧美在线视频免费观看| 色偷偷91综合久久噜噜| 亚洲激情视频在线| 精品久久香蕉国产线看观看亚洲| 国产精自产拍久久久久久| 欧美中文字幕视频在线观看| 成人黄色片网站| 欧美电影免费观看大全| 欧美另类交人妖| 日韩av在线一区二区| 亚洲美女性生活视频| 久久久久久97| 国产精品久久久久久婷婷天堂| 国产精品永久免费视频| 午夜精品久久久99热福利| 欧美重口另类videos人妖| 97视频在线观看成人| 中文字幕免费国产精品| 亚洲91精品在线观看| 国产激情久久久久| 欧美日韩人人澡狠狠躁视频| 国产精品a久久久久久| 亚洲精品国产精品国产自| 国产精品久久久久av免费| 韩国精品久久久999| 久久久久国产精品一区| 亚洲国产天堂网精品网站| 久久久久久久一| 亚洲男人第一av网站| 日韩中文字幕在线播放| 欧美午夜精品久久久久久久| 黑人精品xxx一区一二区| 九色精品美女在线| 欧美高清视频一区二区| 国产午夜精品全部视频在线播放| 欧美成人在线免费| 成人羞羞国产免费| 欧美中文字幕第一页| 国产成人精品免高潮在线观看| 亚洲a在线播放| 久久五月天色综合| 亚洲欧美国产va在线影院| 欧美大胆a视频| 国产一区二区丝袜| 精品久久久国产精品999| 91精品国产乱码久久久久久蜜臀| 狠狠色香婷婷久久亚洲精品| 国产欧美日韩免费看aⅴ视频| 亚洲精品视频二区| 国产一区二区三区在线免费观看| 欧美极品美女视频网站在线观看免费| 欧美午夜丰满在线18影院| 2021国产精品视频| 欧美放荡办公室videos4k| 国产免费一区二区三区香蕉精| 国产精品丝袜一区二区三区| 欧美在线性视频| 亚洲性日韩精品一区二区| 国产性猛交xxxx免费看久久| 不卡av在线网站| 青青久久aⅴ北条麻妃| 91精品国产91久久久久久最新| 国产91网红主播在线观看| 日韩精品中文字幕视频在线| 午夜精品一区二区三区在线| 国产一区二区视频在线观看| 少妇精69xxtheporn| 国产精品高清在线| 久久综合88中文色鬼| 欧美性xxxxx极品娇小| 国产精品久久久久久久av大片| 精品小视频在线| 亚洲欧洲偷拍精品| 国产欧美va欧美va香蕉在线| 欧美日韩国产精品| 热门国产精品亚洲第一区在线| 欧美高清videos高潮hd| 国产精品99久久久久久白浆小说| 欧美在线视频一区| 久久视频在线观看免费| 亚洲天堂av在线播放| 日韩免费观看在线观看| 国产精品都在这里| 亚洲国产女人aaa毛片在线| 亚洲福利视频网站| 这里只有精品久久| 久久久久久亚洲精品| 日本韩国欧美精品大片卡二| 日本韩国在线不卡| 日韩久久精品成人| 国语自产精品视频在线看抢先版图片| 中文字幕av一区二区三区谷原希美| 欧美日韩亚洲高清| 亚洲男子天堂网| 97激碰免费视频| 91在线精品播放| 92福利视频午夜1000合集在线观看| 亚洲一区二区三区香蕉| 日韩成人在线视频| 黑人极品videos精品欧美裸| 日韩一区二区欧美| 欧美日韩日本国产|