十年网站开发经验 + 多家企业客户 + 靠谱的建站团队
量身定制 + 运营维护+专业推广+无忧售后,网站问题一站解决
本篇文章为大家展示了iBATIS中怎么操作CLOB字段,内容简明扼要并且容易理解,绝对能使你眼前一亮,通过这篇文章的详细介绍希望你能有所收获。
创新互联建站服务项目包括善左网站建设、善左网站制作、善左网页制作以及善左网络营销策划等。多年来,我们专注于互联网行业,利用自身积累的技术优势、行业经验、深度合作伙伴关系等,向广大中小型企业、政府机构等提供互联网行业的解决方案,善左网站推广取得了明显的社会效益与经济效益。目前,我们服务的客户以成都为中心已经辐射到善左省份的部分城市,未来相信会继续扩大服务区域并继续获得客户的支持与信任!
环境:
JDK/JRE:1.5.0;Spring:1.2.7;iBATIS:2.1.7;Oracle:9.2.0.1;JDBC:OJDBC14_10g.jar
目标:
基于Spring提供的DBCP,使用iBATIS SQL Maps更新数据库的CLOB字段。
现象:
总是有少部分特定的数据没有更新,且并不是因为数据过长而失败,有时很短的数据也会失败。现象可以重现。原来的SQL的 Maps配置文件如下:
﹤update id="updateByFoodID" parameterClass="Food"﹥ update TB_FOOD ﹤dynamic prepend="set"﹥ ﹤isNotNull prepend="," property="foodDesc"﹥ FOOD_DESC = #foodDesc# ﹤/isNotNull﹥ ﹤isNotNull prepend="," property="foodImage"﹥ FOOD_IMAGE = #foodImage# ﹤/isNotNull﹥ ﹤/dynamic﹥ where FOOD_ID = #foodID# ﹤/update﹥
查看log4j的DEBUG输出,只有Preparing Statement、Executing Statement、和Parameters,并没有异常输出。
iBATIS操作CLOB字段初步分析原因,有如下6个:
1、首先还是怀疑有地方数据过长;
2、操作的长字符串是JSON格式的,并且数据内容可能包含‘?’、‘/’等特殊字符;
3、由于有多个程序同时使用该表,可能是事务隔离的原因;
4、LOB字段在Dynamic Mapped Statements中运行的问题;
5、这个版本的Spring对iBATIS的支持问题;
6、JDBC的版本问题。
iBATIS操作CLOB字段分别分析:
1、VARCHAR2类型的列最长为4000字节,VARCHAR2类型的PL/SQL变量最长为32767字节,操作的数据确实有可能超过64k,但程序中没有使用VARCHAR2变量存储数据,LOB类型的列更是可以存储4G的数据;
2、走查更新失败的字符串内容,没有发现特殊之处;
3、新建一张临时表,用以进行insert和update操作,发现并没有解决问题;
4、去掉动态部分,写最简单的SQL语句,也没有解决问题;
5、(暂时略过);
6、这个可能性比较小,一直都在用***版本的JDBC。
网上有人提到3种办法,其中“第2种”,给SqlMapClientFactoryBean增加lobHandler属性,据称是指能工作在native的驱动方式下,经查Spring的API文档,并不准确,摘录文档如下:
While most databases are able to work with DefaultLobHandler, Oracle just accepts Blob/Clob instances created via its own proprietary BLOB/CLOB API, and additionally doesn't accept large streams for PreparedStatement's corresponding setter methods. Therefore, you need to use a strategy like this LobHandler implementation.
Needs to work on a native JDBC Connection, to be able to cast it to oracle.jdbc.OracleConnection. If you pass in Connections from a connection pool (the usual case in a J2EE environment), you need to set an appropriate NativeJdbcExtractor to allow for automatical retrieval of the underlying native JDBC Connection. LobHandler and NativeJdbcExtractor are separate concerns, therefore they are represented by separate strategy interfaces.
Coded via reflection to avoid dependencies on Oracle classes. Even reads in Oracle constants via reflection because of different Oracle drivers (classes12, ojdbc14) having different constant values! As this LobHandler initializes Oracle classes on instantiation, do not define this as eager-initializing singleton if you do not want to depend on the Oracle JAR being in the class path: use "lazy-init=true" to avoid this issue.
因此,修改Spring配置文件:
﹤bean id="jdbcExtractor" class="org.springframework.jdbc.support.nativejdbc.CommonsDbcpNativeJdbcExtractor" /﹥ ﹤bean id="oracleLobHandler" class="org.springframework.jdbc.support.lob.OracleLobHandler" lazy-init="true"﹥ ﹤property name="nativeJdbcExtractor"﹥﹤ref local="jdbcExtractor" /﹥﹤/property﹥ ﹤/bean﹥ ﹤bean id="sqlMapClient" class="org.springframework.orm.ibatis.SqlMapClientFactoryBean"﹥ ﹤property name="configLocation" value="classpath:conf/sqlMapConfig.xml"﹥﹤/property﹥ ﹤property name="dataSource"﹥﹤ref local="aeqdsDataSource" /﹥﹤/property﹥ ﹤property name="lobHandler"﹥﹤ref local="oracleLobHandler" /﹥﹤/property﹥ ﹤/bean﹥
未果。
使用提到的“第3种”方法,这又有2种方式,第1种是全局性的,在SQL Maps配置文件中加入callback方法:
﹤typeHandler jdbcType="BLOB" javaType="[B" callback="org. ringframework.orm.ibatis.support.BlobByteArrayTypeHandler" /﹥ ﹤typeHandler jdbcType="CLOB" javaType="java.lang.String" llback="org.springframework.orm.ibatis.support.ClobStringTypeHandler" /﹥
第2种是在单个的属性上指定typeHandler,如果是select,就可以用:
﹤resultMap id="foodResult" class="Food"﹥ ﹤result property="foodId" column="FOOD_ID"/﹥ ﹤result property="foodDesc" column="FOOD_DESC" typeHandler="org.springframework.orm.ibatis.support.ClobStringTypeHandler"/﹥ ﹤result property="foodImage" column="FOOD_IMAGE" typeHandler="org.springframework.orm.ibatis.support.BlobByteArrayTypeHandler"/﹥ ﹤/resultMap﹥
如果是insert或update,就可以用:
﹤update id="updateByFoodID" parameterClass="Food"﹥ update TB_FOOD set FOOD_DESC = #foodDesc,handler=org.springframework.orm.ibatis.support.ClobStringTypeHandler#, FOOD_IMAGE = #foodImage,handler=org.springframework.orm.ibatis.support.BlobByteArrayTypeHandler# where FOOD_ID = #foodID# ﹤/update﹥
加入这个callback或typeHandler方法后,iBATIS抱怨没见过这个标记,程序不能启动。把iBATIS升级为2.3.0,顺便也把Spring升级为2.0.2,正常启动。但是效果更差,所有数据均不能更新,且日志显示只进行了Prepare Statement,却没有执行。
死马当活马医:
﹤parameterMap id="updateByFoodIDPara" class="Food"﹥ ﹤parameter property="foodDesc" jdbcType="CLOB" javaType="java.lang.String" /﹥ ﹤parameter property="foodImage" jdbcType="BLOB" javaType="[B" /﹥ ﹤parameter property="foodID" jdbcType="VARCHAR" javaType="java.lang.String" /﹥ ﹤/parameterMap﹥ ﹤update id="updateByFoodID" parameterMap="updateByFoodIDPara"﹥ update TB_FOOD set FOOD_DESC = ?, FOOD_IMAGE = ? where FOOD_ID = ? ﹤/update﹥
依旧未果。
使出杀手锏,把update封装在一个存储过程中(其中不需要特别的事务处理),然后将SQL Maps配置文件的对应内容简简单单的修改为:
﹤procedure id="updateByFoodID" parameterClass="Food"﹥ {call pkg_xxx.sp_xxx (#foodID#, #foodDesc#, #foodImage#)} ﹤/procedure﹥
上述内容就是iBATIS中怎么操作CLOB字段,你们学到知识或技能了吗?如果还想学到更多技能或者丰富自己的知识储备,欢迎关注创新互联行业资讯频道。