Oracle数据库mybatis 插入空值时报错(with JdbcType OTHER)的两种解决方案
在使用Mybatis 3时,发现了这个问题,当插入数据时,如果有一个字段为空值时,系统会报异常,导致插入数据失败。
org.springframework.jdbc.UncategorizedsqlException:Error setting null for parameter #6 with JdbcType OTHER . Try setting adifferent JdbcType for this parameter or a different jdbcTypeForNullconfiguration property.
经过查阅各种资料,终于找到了两种解决方案,分别如下,其中第二种方案个人比较喜欢。
第一种方式,如出错信息中提到的,需要在每个数据变量那里设置相应的jdbcType,示例如下(加粗加下划线部分的内容):
1.<insertid=”insertCustomerLog1″parameterType=”com.diyicai.customer.domain.CustomerLog”>
2.insertintocustomer_log
3.(
4.ID,
5.CUSTOMER_SERVICE_USER_NAME,
6.user_name,
7.CONTENT,
8.LOG_FIRST_TYPE,
9.STATUS,
10.LINKED_ID,
11.FEE,
12.ACCOUNT_FIRST_TYPE,
13.ACCOUNT_SECOND_TYPE,
14.ACCOUNT_THIRD_TYPE,
15.LOG_SECOND_TYPE,
16.LOG_IP,
17.MEMO
18.)
19.values
20.(
21.seq_customer_log.nextval,
22.#{customerServiceUserName,jdbcType=VARCHAR},
23.#{username,
24.#{content,
25.#{logFirstType,jdbcType=NUMERIC},
26.#{status,
27.#{linkedId,
28.#{fee,
29.#{accountFirstType,
30.#{accountSecondType,
31.#{accountThirdType,
32.#{logSecondType,
33.#{logIp,
34.#{memo,jdbcType=VARCHAR}
35.)
36.</insert>
这种方案在网上有很多介绍,但是我不满意,因为太麻烦了,要增加很多代码,导致后期的维护困难增加。后来找到以下第二种方案,很不错,推荐一下。
第二种方案:MyBatis-config.xml中设置当JDBC类型为空值时,要指定的值得,默认为OTHER,我们指定为NULL就好了(注意是大写的NULL)。
顺便补习一下MyBatis-config.xml配置说明,赶紧收藏吧。
1.<!–配置设置–>
2.<settings>
3.<!–配置全局性cache的(开/关)default:true–>
4.<settingname=“cacheEnabled”value=“true”/>
5.
6.<!–是否使用懒加载关联对象同hibernate中的延迟加载一样default:true–>
7.<settingname=“lazyLoadingEnabled”value=“true”/>
8.
9.<!–[当对象使用延迟加载时属性的加载取决于能被引用到的那些延迟属性,否则,按需加载(需要的是时候才去加载)]–>
10.<settingname=“aggressiveLazyLoading”value=“true”/>
11.
12.<!–是否允许单条sql返回多个数据集(取决于驱动的兼容性)default:true–>
13.<settingname=“multipleResultSetsEnabled”value=“true”/>
14.
15.<!–是否可以使用列的别名(取决于驱动的兼容性)default:true–>
16.<settingname=“useColumnLabel”value=“true”/>
17.
18.<!–允许JDBC生成主键。需要驱动器支持。如果设为了true,这个设置将强制使用被生成的主键,有一些驱动器不兼容不过仍然可以执行。default:false–>
19.<settingname=“useGeneratedKeys”value=“false”/>
20.
21.<!–指定MyBatis如何自动映射数据基表的列NONE:不隐射 PARTIAL:部分FULL:全部–>
22.<settingname=“autoMappingBehavior”value=“PARTIAL”/>
23.
24.<!–这是默认的执行类型
25.SIMPLE:简单
26.REUSE:执行器可能重复使用preparedstatements语句
27.BATCH:执行器可以重复执行语句和批量更新
28.–>
29.<settingname=“defaultExecutorType”value=“SIMPLE”/>
30.
31.<!–设置驱动等待数据响应的超时数默认没有设置–>
32.<settingname=“defaultStatementTimeout”value=“25000″/>
33.
34.<!–[是否启用行内嵌套语句defaut:false]–>
35.<settingname=“safeRowBoundsEnabled”value=“false”/>
36.
37.<!–[是否启用数据中A_column自动映射到Java类中驼峰命名的属性default:fasle]–>
38.<settingname=“mapUnderscoreToCamelCase”value=“false”/>
39.
40.<!–设置本地缓存范围session:就会有数据的共享statement:语句范围(这样就不会有数据的共享)defalut:session–>
41.<settingname=“localCacheScope”value=“SESSION”/>
42.
43.<!–设置但JDBC类型为空时,某些驱动程序要指定值,default:OTHER–>
44.<settingname=“jdbcTypeForNull”value=“DEFAULT”/>
45.
46.<!–设置触发延迟加载的方法–>
47.<settingname=“lazyLoadTriggerMethods”value=“equals,clone,hashCode,toString”/>
48.
49.</settings>
本文参考资料:http://www.jb51.cc/article/p-scwfoomf-bqx.htmlhttp://www.jb51.cc/article/p-biwfuklo-bqv.html
另外,再补充一点资料,可能更能让我们了解问题的真相:
适配Oracle数据库的时候,mybatis报了Error setting null parameter,bug发现是参数出现了null值,对于Mybatis,如果进行操作的时候,没有指定jdbcType类型的参数,就可能导致问题。
postgresql,MySQL,sqlSERVER都支持JdbcType.NULL类型,oracle是不支持,适配的时候也因为这个问题导致mybatis报错。
比如,之前配置#{submitDate},它会在oracle中报错:Error settingnull parameter
更改成#{submitDate,jdbcType=DATE},注意jdbcType是区分大小写的。