7種獲取MySQL插入數(shù)據(jù)的自增ID方法,你知道他們的優(yōu)劣嗎腰耙?

mysql和oracle插入的時(shí)候有一個(gè)很大的區(qū)別是榛丢,oracle支持序列做id,mysql本身有一個(gè)列可以做自增長字段挺庞,mysql在插入一條數(shù)據(jù)后晰赞,如何能獲得到這個(gè)自增id的值呢?

方法一:是使用last_insert_id

mysql>SELECTLAST_INSERT_ID();

產(chǎn)生的ID 每次連接后保存在服務(wù)器中选侨。這意味著函數(shù)向一個(gè)給定客戶端返回的值是該客戶端產(chǎn)生對影響AUTO_INCREMENT列的最新語句第一個(gè) AUTO_INCREMENT值的宾肺。這個(gè)值不能被其它客戶端影響,即使它們產(chǎn)生它們自己的 AUTO_INCREMENT值侵俗。這個(gè)行為保證了你能夠找回自己的 ID 而不用擔(dān)心其它客戶端的活動锨用,而且不需要加鎖或處理。

每次mysql_query操作在mysql服務(wù)器上可以理解為一次“原子”操作, 寫操作常常需要鎖表的隘谣, 是mysql應(yīng)用服務(wù)器鎖表不是我們的應(yīng)用程序鎖表增拥。

值得注意的是,如你一次插入了多條記錄寻歧,這個(gè)函數(shù)返回的是第一個(gè)記錄的ID值掌栅。

因?yàn)長AST_INSERT_ID是基于Connection的,只要每個(gè)線程都使用獨(dú)立的Connection對象码泛,LAST_INSERT_ID函數(shù) 將返回該Connection對AUTO_INCREMENT列最新的insert or update*作生成的第一個(gè)record的ID猾封。這個(gè)值不能被其它客戶端(Connection)影響,保證了你能夠找回自己的 ID 而不用擔(dān)心其它客戶端的活動噪珊,而且不需要加鎖晌缘。使用單INSERT語句插入多條記錄, LAST_INSERT_ID返回一個(gè)列表。

LAST_INSERT_ID 是與table無關(guān)的痢站,如果向表a插入數(shù)據(jù)后磷箕,再向表b插入數(shù)據(jù),LAST_INSERT_ID會改變阵难。

方法二:是使用max(id)

使用last_insert_id是基礎(chǔ)連接的岳枷,如果換一個(gè)窗口的時(shí)候調(diào)用則會一直返回10

如果不是頻繁的插入我們也可以使用這種方法來獲取返回的id值

selectmax(id)fromuser;

這個(gè)方法的缺點(diǎn)是不適合高并發(fā)。如果同時(shí)插入的時(shí)候返回的值可能不準(zhǔn)確呜叫。

方法三:是創(chuàng)建一個(gè)存儲過程空繁,在存儲過程中調(diào)用先插入再獲取最大值的操作

DELIMITER $$DROPPROCEDUREIFEXISTS`test`$$CREATEDEFINER=`root`@`localhost`PROCEDURE`test`(innamevarchar(100),outoidint)BEGINinsertintouser(loginname)values(name);selectmax(id)fromuserintooid;selectoid;END$$DELIMITER ;calltest('gg',@id);

方法四:使用@@identity

select@@IDENTITY

@@identity是表示的是最近一次向具有identity屬性(即自增列)的表插入數(shù)據(jù)時(shí)對應(yīng)的自增列的值,是系統(tǒng)定 義的全局變量朱庆。一般系統(tǒng)定義的全局變量都是以@@開頭盛泡,用戶自定義變量以@開頭。比如有個(gè)表A椎工,它的自增列是id饭于,當(dāng)向A表插入一行數(shù)據(jù)后蜀踏,如果插入數(shù)據(jù) 后自增列的值自動增加至101,則通過select @@identity得到的值就是101掰吕。使用@@identity的前提是在進(jìn)行insert操作后果覆,執(zhí)行select @@identity的時(shí)候連接沒有關(guān)閉,否則得到的將是NULL值殖熟。

方法五:是使用getGeneratedKeys()

Connection conn = ;Serializable ret =null;PreparedStatement state = .;ResultSet rs=null;try{? ? state.executeUpdate();? ? rs = state.getGeneratedKeys();if(rs.next()) {? ? ? ? ret = (Serializable) rs.getObject(1);? ? }}catch(SQLException e) {}returnret;

總結(jié)一下局待,在mysql中做完插入之后獲取id在高并發(fā)的時(shí)候是很容易出錯的。另外last_insert_id雖然是基于session的但是不知道為什么沒有測試成功菱属。

方法6:selectkey:

其實(shí)在ibtias框架里使用selectkey這個(gè)節(jié)點(diǎn)钳榨,并設(shè)置insert返回值的類型為integer,就可以返回這個(gè)id值纽门。

SelectKey在Mybatis中是為了解決Insert數(shù)據(jù)時(shí)不支持主鍵自動生成的問題薛耻,他可以很隨意的設(shè)置生成主鍵的方式。

不管SelectKey有多好赏陵,盡量不要遇到這種情況吧饼齿,畢竟很麻煩。

SelectKey需要注意order屬性蝙搔,像Mysql一類支持自動增長類型的數(shù)據(jù)庫中缕溉,order需要設(shè)置為after才會取到正確的值。

像Oracle這樣取序列的情況吃型,需要設(shè)置為before证鸥,否則會報(bào)錯。

另外在用Spring管理事務(wù)時(shí)勤晚,SelectKey和插入在同一事務(wù)當(dāng)中枉层,因而Mysql這樣的情況由于數(shù)據(jù)未插入到數(shù)據(jù)庫中,所以是得不到自動增長的Key运翼。取消事務(wù)管理就不會有問題返干。

下面是一個(gè)xml和注解的例子兴枯,SelectKey很簡單血淌,兩個(gè)例子就夠了:

insert into table1 (name) values (#{name})CALL IDENTITY()

上面xml的傳入?yún)?shù)是map,selectKey會將結(jié)果放到入?yún)?shù)map中财剖。用POJO的情況一樣悠夯,但是有一點(diǎn)需要注意的是,keyProperty對應(yīng)的字段在POJO中必須有相應(yīng)的setter方法躺坟,setter的參數(shù)類型還要一致沦补,否則會報(bào)錯。

@Insert("insert into table2 (name) values(#{name})")@SelectKey(statement="call identity()", keyProperty="nameId", before=false, resultType=int.class)int insertTable2(Name name);

上面是注解的形式咪橙。

方法:7:使用<insert中的useGeneratedKeys 和keyProperty 兩個(gè)屬性

1.在Mybatis Mapper文件中添加屬

性“useGeneratedKeys”和“keyProperty”夕膀,其中keyProperty是Java對象的屬性名虚倒,而不是表格的字段名。

insertinto system(name) values(#{name})

2.Mybatis執(zhí)行完插入語句后产舞,自動將自增長值賦值給對象systemBean的屬性id魂奥。因此,可通過systemBean對應(yīng)的getter方法獲纫酌ā耻煤!

intcount = systemService.insert(systemBean);intid= systemBean.getId();//獲取到的即為新插入記錄的ID

# 注意事項(xiàng)

1.Mybatis Mapper 文件中,“useGeneratedKeys”和“keyProperty”必須添加准颓,而且keyProperty一定得和java對象的屬性名稱一直哈蝇,而不是表格的字段名

2.java Dao中的Insert方法,傳遞的參數(shù)必須為java對象攘已,也就是Bean炮赦,而不能是某個(gè)參數(shù)。

3样勃、在進(jìn)行自增設(shè)計(jì)的時(shí)候眼五,要注意某些關(guān)鍵字,像replace into以及INSERT ... ON DUPLICATE KEY UPDATE 等語句會產(chǎn)生的影響

對于mysql的相關(guān)知識彤灶,其實(shí)說多也多看幼,說少也少,但是現(xiàn)在除了在筆試的時(shí)候?qū)憇ql語句之外幌陕,可能更多的是對實(shí)際生產(chǎn)場景的應(yīng)用诵姜,也就是mysql的調(diào)優(yōu),雖然沒有辦法幫大家找實(shí)際的生產(chǎn)場景進(jìn)行真實(shí)的演示搏熄,但是棚唆,為大家找點(diǎn)資料還是沒什么太大的問題的,結(jié)合mysql面試時(shí)常問的調(diào)優(yōu)問題心例,有相應(yīng)的視頻講解宵凌,幫助你更好的理解相應(yīng)的知識點(diǎn),沒辦法啊止后,兄弟就能幫你到這里了

別著急劃走瞎惫,還有驚喜在等你,這些資料是我家里有幾個(gè)小祖宗译株,在準(zhǔn)備面試的時(shí)候瓜喇,給他們搜集和整理的部分資料,也分享給大家歉糜,有需要的朋友乘寒,歡迎關(guān)注+轉(zhuǎn)發(fā),然后私信“資料”即可查看獲取方式

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末匪补,一起剝皮案震驚了整個(gè)濱河市伞辛,隨后出現(xiàn)的幾起案子烂翰,更是在濱河造成了極大的恐慌,老刑警劉巖蚤氏,帶你破解...
    沈念sama閱讀 207,113評論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件刽酱,死亡現(xiàn)場離奇詭異,居然都是意外死亡瞧捌,警方通過查閱死者的電腦和手機(jī)棵里,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評論 2 381
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來姐呐,“玉大人殿怜,你說我怎么就攤上這事∈锷埃” “怎么了头谜?”我有些...
    開封第一講書人閱讀 153,340評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長鸠澈。 經(jīng)常有香客問我柱告,道長,這世上最難降的妖魔是什么笑陈? 我笑而不...
    開封第一講書人閱讀 55,449評論 1 279
  • 正文 為了忘掉前任际度,我火速辦了婚禮,結(jié)果婚禮上涵妥,老公的妹妹穿的比我還像新娘乖菱。我一直安慰自己,他們只是感情好蓬网,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評論 5 374
  • 文/花漫 我一把揭開白布窒所。 她就那樣靜靜地躺著,像睡著了一般帆锋。 火紅的嫁衣襯著肌膚如雪吵取。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,166評論 1 284
  • 那天锯厢,我揣著相機(jī)與錄音皮官,去河邊找鬼。 笑死哲鸳,一個(gè)胖子當(dāng)著我的面吹牛臣疑,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播徙菠,決...
    沈念sama閱讀 38,442評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼郁岩!你這毒婦竟也來了婿奔?” 一聲冷哼從身側(cè)響起缺狠,我...
    開封第一講書人閱讀 37,105評論 0 261
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎萍摊,沒想到半個(gè)月后挤茄,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,601評論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡冰木,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評論 2 325
  • 正文 我和宋清朗相戀三年穷劈,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片踊沸。...
    茶點(diǎn)故事閱讀 38,161評論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡歇终,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出逼龟,到底是詐尸還是另有隱情评凝,我是刑警寧澤,帶...
    沈念sama閱讀 33,792評論 4 323
  • 正文 年R本政府宣布腺律,位于F島的核電站奕短,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏匀钧。R本人自食惡果不足惜翎碑,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評論 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望之斯。 院中可真熱鬧杈女,春花似錦、人聲如沸吊圾。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,352評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽项乒。三九已至啰劲,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間檀何,已是汗流浹背蝇裤。 一陣腳步聲響...
    開封第一講書人閱讀 31,584評論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留频鉴,地道東北人栓辜。 一個(gè)月前我還...
    沈念sama閱讀 45,618評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像垛孔,于是被迫代替她去往敵國和親藕甩。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評論 2 344