mybatis 为什么要设置jdbcType

本文详细介绍了在使用MyBatis框架进行与Oracle数据库交互时,遇到null参数导致的错误,并提供了解决方案。通过指定参数的JdbcType类型,可以避免此类异常,确保数据库操作的顺利执行。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

以前没注意过这个问题,用ibatis的时候从来没有设置过jdbcType。ibatis也不会出现这个问题。学习了

——————————————————————————————————————————————

转载自:http://makemyownlife.iteye.com/blog/1610021

前天遇到一个问题 异常显示如下: 

引用

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: 无效的列类型: 1111 
; uncategorized SQLException for SQL []; SQL state [99999]; error code [17004]; 无效的列类型: 1111; nested exception is java.sql.SQLException: 无效的列类型: 1111 


对应的sqlmap如下: 
Xml代码  收藏代码
  1. <insert id="insertCustomerLog" parameterType="map">  
  2.       insert into customer_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} ,  
  23.             #{username},  
  24.             #{content},  
  25.             #{logFirstType},  
  26.             #{status},  
  27.             #{linkedId},  
  28.             #{fee},  
  29.             #{accountFirstType},  
  30.             #{accountSecondType},  
  31.             #{accountThirdType},  
  32.             #{logSecondType},  
  33.             #{logIp},  
  34.             #{memo}  
  35.       )  
  36.   </insert>  


查询了一下 一些资料说是: 

引用

MyBatis 插入空值时,需要指定JdbcType 
mybatis insert空值报空值异常,但是在pl/sql不会提示错误,主要原因是mybatis无法进行转换, 


所以将xml改为: 
Xml代码  收藏代码
  1. <insert id="insertCustomerLog1" parameterType="com.diyicai.customer.domain.CustomerLog">  
  2.         insert into customer_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,jdbcType=VARCHAR},  
  24.                #{content,jdbcType=VARCHAR},  
  25.                #{logFirstType,jdbcType=NUMERIC},  
  26.                #{status,jdbcType=NUMERIC},  
  27.                #{linkedId,jdbcType=VARCHAR},  
  28.                #{fee,jdbcType=NUMERIC},  
  29.                #{accountFirstType,jdbcType=NUMERIC},  
  30.                #{accountSecondType,jdbcType=NUMERIC},  
  31.                #{accountThirdType,jdbcType=NUMERIC},  
  32.                #{logSecondType,jdbcType=NUMERIC},  
  33.                #{logIp,jdbcType=VARCHAR},  
  34.                #{memo,jdbcType=VARCHAR}  
  35.                )  
  36.     </insert>  


这个时候 运行正常。 

我不甘心 想试试 ibatis2 是否也有同样的问题 

xml如下: 
Xml代码  收藏代码
  1. <insert id="BasicUserInfoDaoImpl.testMap" parameterClass="java.util.HashMap">  
  2.                 insert into customer_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# ,  
  23.                       #username#,  
  24.                       #content#,  
  25.                       #logFirstType#,  
  26.                       #status#,  
  27.                       #linkedId#,  
  28.                       #fee#,  
  29.                       #accountFirstType#,  
  30.                       #accountSecondType#,  
  31.                       #accountThirdType#,  
  32.                       #logSecondType#,  
  33.                       #logIp#,  
  34.                       #memo#  
  35.                 )  
  36.     </insert>  


这个时候 发现 ibatis2 可以正常的执行 数据库可以正常的插入数据 。 
错误日志是在:org.apache.ibatis.type.BaseTypeHandler这个类的第17行打出的。根据异常上面的代码 
Java代码  收藏代码
  1. if (parameter == null) {  
  2.   if (jdbcType == null) {  
  3.   try {  
  4.   ps.setNull(i, JdbcType.OTHER.TYPE_CODE);  
  5.   } catch (SQLException e) {  
  6.   throw new TypeException("Error setting null parameter. Most JDBC drivers require that the JdbcType must be specified for all nullable parameters. Cause: " + e, e);  
  7.   }  
  8.   } else {  
  9.   ps.setNull(i, jdbcType.TYPE_CODE);  
  10.   }  
  11.   } else {  
  12.   setNonNullParameter(ps, i, parameter, jdbcType);  
  13.   }  

可以看出,是因为你传入的参数的字段为null对象无法获取对应的jdbcType类型,而报的错误。 
你只要在insert语句中insert的对象加上jdbcType就可以了,修改如下: 
#{menuTitle,jdbcType=VARCHAR} 
这样就可以解决以上错误了。 
看来需要真正的了解mybatis 和ibatis 呀。 
希望遇到同样问题的朋友一起交流。 


晚上看到了http://wksandy.iteye.com/blog/1443133 感谢 感谢 WKsandy ,他的文字写得很好 


引用
还有在向oracle插入数据时,mybatis3报Error setting null parameter. Most JDBC drivers require that the JdbcType must be specified for all nullable parameters,是由于参数出现了null值,对于Mybatis,如果进行操作的时候,没有指定jdbcType类型的参数,mybatis默认jdbcType.OTHER导致,给参数加上jdbcType可解决(注意大小写) 

http://code.google.com/p/mybatis/issues/detail?id=224&q=Error%20setting%20null%20parameter&colspec=ID
### MyBatis 中 `JdbcType` 的用法 在 MyBatis 映射文件中,`JdbcType` 是用于指定 Java 类型和数据库 JDBC 类型之间转换的关键属性。当处理可能为空的结果集字段时,设置 `JdbcType` 可以帮助防止潜在的异常。 #### 基本概念 MyBatis 使用 `JdbcType` 来定义如何将 SQL 查询结果中的列映射到 Java 对象的属性上。对于某些特定类型的字段(如日期、布尔值),显式声明 `JdbcType` 能够提高数据一致性和可靠性[^1]。 #### 配置方式 可以通过 XML 或者注解的方式配置 `JdbcType`: - **XML 方式** ```xml <resultMap id="userResultMap" type="User"> <id property="id" column="USER_ID" jdbcType="INTEGER"/> <result property="name" column="USERNAME" jdbcType="VARCHAR"/> </resultMap> ``` 在这个例子中,`jdbcType` 属性指定了每一列对应的 JDBC 数据类型,这有助于确保即使遇到 NULL 值也能正确解析并赋给相应的对象属性。 - **注解方式** ```java @Results({ @Result(property = "id", column = "USER_ID", jdbcType=JdbcType.INTEGER), @Result(property = "name", column = "USERNAME", jdbcType=JdbcType.VARCHAR) }) public List<User> selectUsers(); ``` 这里展示了通过注解来设定 `JdbcType` 的方法,适用于更简洁的应用场景下。 #### 处理空值的情况 为了更好地管理数据库中的 NULL 值,在 resultMap 定义里加入 `nullValue` 和 `typeHandler` 参数是非常有用的实践之一。例如: ```xml <result property="emailVerified" column="EMAIL_VERIFIED" jdbcType="BIT" nullValue="false" typeHandler="org.example.BooleanTypeHandler"/> ``` 这段代码片段说明了怎样利用自定义的 Type Handler 来控制 NULL 到默认值之间的转换逻辑。 #### 批量更新操作注意事项 需要注意的是,当执行批量更新语句时,只有当 ExecutorType 设置为 BATCH 模式时才会生效存储于 JDBC Driver 内部的一系列待提交命令,并可通过调用 flushStatements 方法强制刷新这些缓存的操作[^3]。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值