關(guān)于Mybatis插入對(duì)象時(shí)空值的處理
Mybatis插入對(duì)象時(shí)空值
Mybatis中經(jīng)常會(huì)有插入數(shù)據(jù)的情景,有時(shí)傳輸?shù)膶?duì)象字段并不是完整的,如果不做任何處理則會(huì)拋出異常,影響程序執(zhí)行。
如存在對(duì)象賦值不完整的情況可以在字段后添加 jdbcType 如:
INSERT INTO student(`uid`,`name`,`class`) VALUES(#{uid,jdbcType=VARCHAR},#{name,jdbcType=VARCHAR},#{class,jdbcType=VARCHAR})
然后再 mybatis-config.xml 中添加配置
<settings> ?? ?<setting name="jdbcTypeForNull" value="NULL" /> ? </settings>
如若是其它類型也可使用 <if> 標(biāo)簽
<if test="otherType != null and otherType!= ''" > ? ? otherType = #{otherType} </if>
這樣在數(shù)據(jù)庫(kù)未做限制時(shí),就不會(huì)影響到數(shù)據(jù)的添加或修改。
需要注意的點(diǎn)MyBatis插入空值時(shí),需要指定JdbcType
前天遇到一個(gè)問題 異常顯示如下:
Exception in thread "main" org.springframework.jdbc.UncategorizedSQLException: Error setting null for parameter #6 with JdbcType OTHER . Try setting a different JdbcType for this parameter or a different jdbcTypeForNull configuration property. Cause: java.sql.SQLException: 無(wú)效的列類型: 1111
; uncategorized SQLException for SQL []; SQL state [99999]; error code [17004]; 無(wú)效的列類型: 1111; nested exception is java.sql.SQLException: 無(wú)效的列類型: 1111
對(duì)應(yīng)的sqlmap如下:
<insert id="insertCustomerLog" parameterType="map"> ? ? ? ? insert into customer_log ? ? ? ? ( ? ? ? ? ID, ? ? ? ? CUSTOMER_SERVICE_USER_NAME, ? ? ? ? user_name, ? ? ? ? CONTENT, ? ? ? ? LOG_FIRST_TYPE, ? ? ? ? STATUS, ? ? ? ? LINKED_ID, ? ? ? ? FEE, ? ? ? ? ACCOUNT_FIRST_TYPE, ? ? ? ? ACCOUNT_SECOND_TYPE, ? ? ? ? ACCOUNT_THIRD_TYPE, ? ? ? ? LOG_SECOND_TYPE, ? ? ? ? LOG_IP, ? ? ? ? MEMO ? ? ? ? ) ? ? ? ? values ? ? ? ? ( ? ? ? ? ? ? ? ?seq_customer_log.nextval , ? ? ? ? ? ? ? #{customerServiceUserName} , ? ? ? ? ? ? ? #{username}, ? ? ? ? ? ? ? #{content}, ? ? ? ? ? ? ? #{logFirstType}, ? ? ? ? ? ? ? #{status}, ? ? ? ? ? ? ? #{linkedId}, ? ? ? ? ? ? ? #{fee}, ? ? ? ? ? ? ? #{accountFirstType}, ? ? ? ? ? ? ? #{accountSecondType}, ? ? ? ? ? ? ? #{accountThirdType}, ? ? ? ? ? ? ? #{logSecondType}, ? ? ? ? ? ? ? #{logIp}, ? ? ? ? ? ? ? #{memo} ? ? ? ? ) ? ? </insert>
查詢了一下 一些資料說是
MyBatis 插入空值時(shí),需要指定JdbcType ,mybatis insert空值報(bào)空值異常,但是在pl/sql不會(huì)提示錯(cuò)誤,主要原因是mybatis無(wú)法進(jìn)行轉(zhuǎn)換。
<insert id="insertCustomerLog1" parameterType="com.diyicai.customer.domain.CustomerLog"> ? ? ? ? ? insert into customer_log ? ? ? ? ? ? ? ? ?( ? ? ? ? ? ? ? ? ?ID, ? ? ? ? ? ? ? ? ?CUSTOMER_SERVICE_USER_NAME, ? ? ? ? ? ? ? ? ?user_name , ? ? ? ? ? ? ? ? ?CONTENT, ? ? ? ? ? ? ? ? ?LOG_FIRST_TYPE, ? ? ? ? ? ? ? ? ?STATUS, ? ? ? ? ? ? ? ? ?LINKED_ID, ? ? ? ? ? ? ? ? ?FEE, ? ? ? ? ? ? ? ? ?ACCOUNT_FIRST_TYPE, ? ? ? ? ? ? ? ? ?ACCOUNT_SECOND_TYPE, ? ? ? ? ? ? ? ? ?ACCOUNT_THIRD_TYPE, ? ? ? ? ? ? ? ? ?LOG_SECOND_TYPE, ? ? ? ? ? ? ? ? ?LOG_IP, ? ? ? ? ? ? ? ? ?MEMO ? ? ? ? ? ? ? ? ?) ? ? ? ? ? ? ? ? ?values ? ? ? ? ? ? ? ? ?( ? ? ? ? ? ? ? ? ?seq_customer_log.nextval , ? ? ? ? ? ? ? ? ?#{customerServiceUserName,jdbcType=VARCHAR} , ? ? ? ? ? ? ? ? ?#{username,jdbcType=VARCHAR}, ? ? ? ? ? ? ? ? ?#{content,jdbcType=VARCHAR}, ? ? ? ? ? ? ? ? ?#{logFirstType,jdbcType=NUMERIC}, ? ? ? ? ? ? ? ? ?#{status,jdbcType=NUMERIC}, ? ? ? ? ? ? ? ? ?#{linkedId,jdbcType=VARCHAR}, ? ? ? ? ? ? ? ? ?#{fee,jdbcType=NUMERIC}, ? ? ? ? ? ? ? ? ?#{accountFirstType,jdbcType=NUMERIC}, ? ? ? ? ? ? ? ? ?#{accountSecondType,jdbcType=NUMERIC}, ? ? ? ? ? ? ? ? ?#{accountThirdType,jdbcType=NUMERIC}, ? ? ? ? ? ? ? ? ?#{logSecondType,jdbcType=NUMERIC}, ? ? ? ? ? ? ? ? ?#{logIp,jdbcType=VARCHAR}, ? ? ? ? ? ? ? ? ?#{memo,jdbcType=VARCHAR} ? ? ? ? ? ? ? ? ?) ? ? ? </insert> ?
錯(cuò)誤日志是在:org.apache.ibatis.type.BaseTypeHandler這個(gè)類的第17行打出的。根據(jù)異常上面的代碼 :
if (parameter == null) { ? ? if (jdbcType == null) { ? ? try { ? ? ps.setNull(i, JdbcType.OTHER.TYPE_CODE); ? ? } catch (SQLException e) { ? ? throw new TypeException("Error setting null parameter. Most JDBC drivers require that the JdbcType must be specified for all nullable parameters. Cause: " + e, e); ? ? } ? ? } else { ? ? ps.setNull(i, jdbcType.TYPE_CODE); ? ? } ? ? } else { ? ? setNonNullParameter(ps, i, parameter, jdbcType); ? ? } ?
可以看出,是因?yàn)槟銈魅氲膮?shù)的字段為null對(duì)象無(wú)法獲取對(duì)應(yīng)的jdbcType類型,而報(bào)的錯(cuò)誤。 你只要在insert語(yǔ)句中insert的對(duì)象加上jdbcType就可以了,修改如下: #{menuTitle,jdbcType=VARCHAR} ,這樣就可以解決以上錯(cuò)誤了。
但是,如果我們?yōu)槊總€(gè)sql都指定jdbc類型,也比較麻煩,可以mybatis-config.xml種全局設(shè)置下:
<settings> ?? ?<setting name="jdbcTypeForNull" value="NULL"/> </settings>
另外,再補(bǔ)充一點(diǎn)資料,可能更能讓我們了解問題的真相:
適配oracle數(shù)據(jù)庫(kù)的時(shí)候,mybatis報(bào)了Error setting null parameter,bug發(fā)現(xiàn)是參數(shù)出現(xiàn)了null值,對(duì)于Mybatis,如果進(jìn)行操作的時(shí)候,沒有指定jdbcType類型的參數(shù),就可能導(dǎo)致問題。
postgreSQL,MySQL,SQLSERVER都支持JdbcType.NULL類型,Oracle是不支持,適配的時(shí)候也因?yàn)檫@個(gè)問題導(dǎo)致mybatis報(bào)錯(cuò)。
比如,之前配置#{submitDate},它會(huì)在oracle中報(bào)錯(cuò):Error settingnull parameter
更改成#{submitDate,jdbcType=DATE},注意jdbcType是區(qū)分大小寫的。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
SpringBoot搭配AOP實(shí)現(xiàn)自定義注解
這篇文章主要為大家詳細(xì)介紹了SpringBoot如何搭配AOP實(shí)現(xiàn)自定義注解,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2022-12-12Java單例模式實(shí)現(xiàn)靜態(tài)內(nèi)部類方法示例
這篇文章主要介紹了Java單例模式實(shí)現(xiàn)靜態(tài)內(nèi)部類方法示例,涉及構(gòu)造函數(shù)私有化等相關(guān)內(nèi)容,需要的朋友可以了解下。2017-09-09java實(shí)現(xiàn)簡(jiǎn)易點(diǎn)菜器
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)簡(jiǎn)易點(diǎn)菜器,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-12-12