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ā),然后私信“資料”即可查看獲取方式