MYSQL數(shù)據(jù)插入之返回自增主鍵ID的方法詳解
前言
我們都知道,mysql中的insert插入之后會(huì)有返回值,返回的是影響的行數(shù),也就是說(shuō),成功插入一條數(shù)據(jù)之后返回的是1,失敗則返回0。那么,很多時(shí)候我們都想要得到最后插入的id值,下面七種方法均可,結(jié)合場(chǎng)景使用。
MYSQL插入數(shù)據(jù)后返回自增ID的方法
mysql和oracle插入的時(shí)候有一個(gè)很大的區(qū)別是,oracle支持序列做id,mysql本身有一個(gè)列可以做自增長(zhǎng)字段,mysql在插入一條數(shù)據(jù)后,如何能獲得到這個(gè)自增id的值呢?
1、使用last_insert_id
SELECT LAST_INSERT_ID();
產(chǎn)生的ID 每次連接后保存在服務(wù)器中。這意味著函數(shù)向一個(gè)給定客戶端返回的值是該客戶端產(chǎn)生對(duì)影響AUTO_INCREMENT列的最新語(yǔ)句第一個(gè) AUTO_INCREMENT值的。這個(gè)值不能被其它客戶端影響,即使它們產(chǎn)生它們自己的 AUTO_INCREMENT值。這個(gè)行為保證了你能夠找回自己的 ID 而不用擔(dān)心其它客戶端的活動(dòng),而且不需要加鎖或處理。 每次mysql_query操作在mysql服務(wù)器上可以理解為一次“原子”操作, 寫操作常常需要鎖表的, 是mysql應(yīng)用服務(wù)器鎖表不是我們的應(yīng)用程序鎖表。
值得注意的是,如果你一次插入了多條記錄,這個(gè)函數(shù)返回的是第一個(gè)記錄的ID值。
因?yàn)長(zhǎng)AST_INSERT_ID是基于Connection的,只要每個(gè)線程都使用獨(dú)立的Connection對(duì)象,LAST_INSERT_ID函數(shù) 將返回該Connection對(duì)AUTO_INCREMENT列最新的insert or update*作生成的第一個(gè)record的ID。這個(gè)值不能被其它客戶端(Connection)影響,保證了你能夠找回自己的 ID 而不用擔(dān)心其它客戶端的活動(dòng),而且不需要加鎖。使用單INSERT語(yǔ)句插入多條記錄, LAST_INSERT_ID返回一個(gè)列表。
LAST_INSERT_ID 是與table無(wú)關(guān)的,如果向表a插入數(shù)據(jù)后,再向表b插入數(shù)據(jù),LAST_INSERT_ID會(huì)改變。 所以我們可以使用下面方式:
<insert id="insert" parameterType="com.dl.blog.pojo.BlogTag">
<selectKey resultType="integer" order="AFTER" keyProperty="id">
SELECT LAST_INSERT_ID()
</selectKey>
insert into blog_tag (name)
values (#{name,jdbcType=VARCHAR})
</insert>2、使用max(id)
使用last_insert_id是基礎(chǔ)連接的,如果換一個(gè)窗口的時(shí)候調(diào)用則會(huì)一直返回10 如果不是頻繁的插入我們也可以使用這種方法來(lái)獲取返回的id值
select max(id) from user;
這個(gè)方法的缺點(diǎn)是不適合高并發(fā)。如果同時(shí)插入的時(shí)候返回的值可能不準(zhǔn)確。
3、獲取最大值
創(chuàng)建一個(gè)存儲(chǔ)過(guò)程,在存儲(chǔ)過(guò)程中調(diào)用先插入再獲取最大值的操作
DELIMITER $$
DROP PROCEDURE IF EXISTS `test` $$
CREATE DEFINER=`root`@`localhost` PROCEDURE `test`(in name varchar(100),out oid int)
BEGIN
insert into user(loginname) values(name);
select max(id) from user into oid;
select oid;
END $$
DELIMITER ;
call test('gg',@id);4、使用@@identity
select @@IDENTITY
@@identity是表示的是最近一次向具有identity屬性(即自增列)的表插入數(shù)據(jù)時(shí)對(duì)應(yīng)的自增列的值,是系統(tǒng)定 義的全局變量。一般系統(tǒng)定義的全局變量都是以@@開(kāi)頭,用戶自定義變量以@開(kāi)頭。比如有個(gè)表A,它的自增列是id,當(dāng)向A表插入一行數(shù)據(jù)后,如果插入數(shù)據(jù) 后自增列的值自動(dòng)增加至101,則通過(guò)select @@identity得到的值就是101。使用@@identity的前提是在進(jìn)行insert操作后,執(zhí)行select @@identity的時(shí)候連接沒(méi)有關(guān)閉,否則得到的將是NULL值。
5、使用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) {
}
return ret;總結(jié)一下,在mysql中做完插入之后獲取id在高并發(fā)的時(shí)候是很容易出錯(cuò)的。另外last_insert_id雖然是基于session的但是不知道為什么沒(méi)有測(cè)試成功。
6、selectkey
其實(shí)在ibtias框架里使用selectkey這個(gè)節(jié)點(diǎn),并設(shè)置insert返回值的類型為integer,就可以返回這個(gè)id值。
SelectKey在Mybatis中是為了解決Insert數(shù)據(jù)時(shí)不支持主鍵自動(dòng)生成的問(wèn)題,他可以很隨意的設(shè)置生成主鍵的方式。
不管SelectKey有多好,盡量不要遇到這種情況吧,畢竟很麻煩。
selectKey Attributes
| 屬性 | 描述 |
| keyProperty | selectKey 語(yǔ)句結(jié)果應(yīng)該被設(shè)置的目標(biāo)屬性。 |
| resultType | 結(jié)果的類型。MyBatis 通常可以算出來(lái),但是寫上也沒(méi)有問(wèn)題。MyBatis 允許任何簡(jiǎn)單類型用作主鍵的類型,包括字符串。 |
| order | 這可以被設(shè)置為 BEFORE 或 AFTER。如果設(shè)置為 BEFORE,那么它會(huì)首先選擇主鍵,設(shè)置 keyProperty 然后執(zhí)行插入語(yǔ)句。如果設(shè)置為 AFTER,那么先執(zhí)行插入語(yǔ)句,然后是 selectKey 元素-這和如 Oracle 數(shù)據(jù)庫(kù)相似,可以在插入語(yǔ)句中嵌入序列調(diào)用。 |
| statementType | 和前面的相 同,MyBatis 支持 STATEMENT ,PREPARED 和CALLABLE 語(yǔ)句的映射類型,分別代表 PreparedStatement 和CallableStatement 類型。 |
SelectKey需要注意order屬性,像Mysql一類支持自動(dòng)增長(zhǎng)類型的數(shù)據(jù)庫(kù)中,order需要設(shè)置為after才會(huì)取到正確的值。
像Oracle這樣取序列的情況,需要設(shè)置為before,否則會(huì)報(bào)錯(cuò)。
另外在用Spring管理事務(wù)時(shí),SelectKey和插入在同一事務(wù)當(dāng)中,因而Mysql這樣的情況由于數(shù)據(jù)未插入到數(shù)據(jù)庫(kù)中,所以是得不到自動(dòng)增長(zhǎng)的Key。取消事務(wù)管理就不會(huì)有問(wèn)題。 下面是一個(gè)xml和注解的例子,SelectKey很簡(jiǎn)單,兩個(gè)例子就夠了:
<insert id="insert" parameterType="map">
insert into table1 (name) values (#{name})
<selectKey resultType="java.lang.Integer" keyProperty="id">
CALL IDENTITY()
</selectKey>
</insert>上面xml的傳入?yún)?shù)是map,selectKey會(huì)將結(jié)果放到入?yún)?shù)map中。用POJO的情況一樣,但是有一點(diǎn)需要注意的是,keyProperty對(duì)應(yīng)的字段在POJO中必須有相應(yīng)的setter方法,setter的參數(shù)類型還要一致,否則會(huì)報(bào)錯(cuò)。 注解版:
@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è)屬性
首先在Mybatis Mapper文件中添加屬性 useGeneratedKeys 和 keyProperty ,其中keyProperty是Java對(duì)象的屬性名,而不是表格的字段名。
<insert id="insert" parameterType="com.dl.blog.pojo.BlogTag" useGeneratedKeys="true" keyProperty="id">
insert into blog_tag (name)
values (#{name,jdbcType=VARCHAR})
</insert>然后我們?cè)趈ava代碼中Mybatis執(zhí)行完插入語(yǔ)句后,自動(dòng)將自增長(zhǎng)值賦值給對(duì)象systemBean的屬性id。因此,可通過(guò)systemBean對(duì)應(yīng)的getter方法獲??!
int count=blogTagMapper.insert(blogTag);//返回的是影響記錄的行數(shù) int id=blogTag.getId();//獲取到的即為新插入記錄的ID
POJO實(shí)體類,省略getter、setter、和構(gòu)造方法
public class BlogTag {
private Integer id;
private String name;
}數(shù)據(jù)庫(kù)表結(jié)構(gòu):

【注意事項(xiàng)】
1.Mybatis Mapper 文件中,useGeneratedKeys和keyProperty必須添加,而且keyProperty一定得和java對(duì)象的屬性名稱一直,而不是表格的字段名。
2.java Dao中的Insert方法,傳遞的參數(shù)必須為java對(duì)象,也就是Bean,而不能是某個(gè)參數(shù)。
到此這篇關(guān)于MYSQL數(shù)據(jù)插入之返回自增主鍵ID的方法詳解的文章就介紹到這了,更多相關(guān)MYSQL返回自增主鍵ID內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
淺談mysql 系統(tǒng)用戶最大文件打開(kāi)數(shù)限制
這篇文章主要介紹了mysql 系統(tǒng)用戶最大文件打開(kāi)數(shù)限制,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03
MySQL定時(shí)任務(wù)EVENT事件的使用方法
本文主要介紹了MySQL定時(shí)任務(wù)EVENT事件的使用方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2023-05-05
mysql 遠(yuǎn)程連接數(shù)據(jù)庫(kù)的方法集合
MySQL數(shù)據(jù)庫(kù)不允許從遠(yuǎn)程訪問(wèn)怎么辦?本文提供了三種解決方法:2008-03-03
MySql報(bào)錯(cuò):Duplicate entry ‘10‘ for key&nbs
經(jīng)常遇到這個(gè)問(wèn)題,今天我把這個(gè)問(wèn)題記錄下來(lái),以后如果有其他人遇到相同的問(wèn)題,不至于困惑,在操作數(shù)據(jù)庫(kù)的時(shí)候,經(jīng)常會(huì)出現(xiàn)Duplicate entry ‘10’ for key 'PRIMARY’這樣的報(bào)錯(cuò),本文將給大家講講如何解決這個(gè)問(wèn)題,需要的朋友可以參考下2023-10-10
mysql5.7.18安裝時(shí)提示無(wú)法找到入口問(wèn)題的解決方法
這篇文章主要為大家詳細(xì)介紹了mysql5.7.18安裝時(shí)出現(xiàn)無(wú)法找到入口問(wèn)題的解決方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-04-04
MySQL版本低了不支持兩個(gè)時(shí)間戳類型的值解決方法
在本篇文章里小編給大家分享了關(guān)于MySQL 版本低了,不支持兩個(gè)時(shí)間戳類型的值的相關(guān)知識(shí)點(diǎn),有興趣的朋友們可以參考下。2019-09-09
MySQL函數(shù)與存儲(chǔ)過(guò)程字符串長(zhǎng)度限制的解決
本文主要介紹了MySQL函數(shù)與存儲(chǔ)過(guò)程字符串長(zhǎng)度限制的解決,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-08-08

