MyBatis虽然有很好的SQL执行性能,但毕竟不是完整的ORM框架,不同的数据库之间SQL执行还是有差异。

  笔者最近在升级 Oracle 驱动至 ojdbc 7 ,就发现了处理DATE类型存在问题。还好MyBatis提供了使用自定义TypeHandler转换类型的功能。

  本文介绍如下使用 TypeHandler 实现日期类型的转换。

  问题背景

  项目中有如下的字段,是采用的DATE类型:

  birthday = #{birthday, jdbcType=DATE},

  在更新 Oracle 驱动之前,DateOnlyTypeHandler会做出处理,将 jdbcType 是 DATE 的数据转为短日期格式(‘年月日’)插入数据库。毕竟是生日嘛,只需要精确到年月日即可。

  但是,升级 Oracle 驱动至 ojdbc 7 ,就发现了处理DATE类型存在问题。插入的数据格式变成了长日期格式(‘年月日时分秒’),显然不符合需求了。

  解决方案:

  MyBatis提供了使用自定义TypeHandler转换类型的功能。可以自己写个TypeHandler来对 DATE 类型做特殊处理:

  /**

  * Welcome to https://waylau.com

  */

  package com.waylau.lite.mall.type;

  import java.sql.CallableStatement;

  import java.sql.PreparedStatement;

  import java.sql.ResultSet;

  import java.sql.SQLException;

  import java.text.DateFormat;

  import java.util.Date;

  import org.apache.ibatis.type.BaseTypeHandler;

  import org.apache.ibatis.type.JdbcType;

  import org.apache.ibatis.type.MappedJdbcTypes;

  import org.apache.ibatis.type.MappedTypes;

  /**

  * 自定义TypeHandler,用于将日期转为'yyyy-MM-dd'

  *

  * @since 1.0.0 2018年10月10日

  * @author Way Lau

  */

  @MappedJdbcTypes(JdbcType.DATE)

  @MappedTypes(Date.class)

  public class DateShortTypeHandler extends BaseTypeHandler{

  @Override

  public void setNonNullParameter(PreparedStatement ps, int i, Date parameter, JdbcType jdbcType)

  throws SQLException {

  DateFormat df = DateFormat.getDateInstance();

  String dateStr = df.format(parameter);

  ps.setDate(i, java.sql.Date.valueOf(dateStr));

  }

  @Override

  public Date getNullableResult(ResultSet rs, String columnName) throws SQLException {

  java.sql.Date sqlDate = rs.getDate(columnName);

  if (sqlDate != null) {

  return new Date(sqlDate.getTime());

  }

  return null;

  }

  @Override

  public Date getNullableResult(ResultSet rs, int columnIndex) throws SQLException {

  java.sql.Date sqlDate = rs.getDate(columnIndex);

  if (sqlDate != null) {

  return new Date(sqlDate.getTime());

  }

  return null;

  }

  @Override

  public Date getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {

  java.sql.Date sqlDate = cs.getDate(columnIndex);

  if (sqlDate != null) {

  return new Date(sqlDate.getTime());

  }

  return null;

  }

  }

  如果是 Spring 项目,以下面方式进行 TypeHandler 的配置:

  !-- 自定义 --

  !--声明TypeHandler bean--

  bean id=dateShortTypeHandler class="com".waylau.lite.mall.type.DateShortTypeHandler/

  !-- MyBatis 工厂 --

  bean id=sqlSessionFactory class="org".mybatis.spring.SqlSessionFactoryBean

  property name=dataSource ref=dataSource /

  !--TypeHandler注入--

  property name=typeHandlers ref=dateShortTypeHandler/

  /bean

  如何使用 TypeHandler

  方式1 :指定 jdbcType 为 DATE

  比如,目前,项目中有如下的字段,是采用的DATE类型:

  birthday = #{birthday, jdbcType=DATE},

  方式2 :指定 typeHandler

  指定 typeHandler 为我们自定义的 TypeHandler:

  birthday = #{birthday, typeHandler=com.waylau.lite.mall.type.DateShortT

MyBatis使用自定义TypeHandler转换类型的更多相关文章

  1. MyBatis使用自定义TypeHandler转换类型的实现方法

    From: http://www.manongjc.com/article/15577.html 这篇文章主要介绍了MyBatis使用自定义TypeHandler转换类型的实现方法,本文介绍使用Typ ...

  2. mybatis实现自定义typeHandler

    java8以前的日期处理有多恶心,相信不少人都深有体会.与mabatis集成查询数据库中的日期字段映射为java日期类型或者字符型的时候,会多出一个".0".当然可以自行处理.但是 ...

  3. mybatis类型转换器 - 自定义全局转换enum

    在数据模型.接口参数等场景部分属性参数为一些常量值,比如性别:男.女.若是定义成int或String类型,于是类型本身的范围太宽,要求使用者需要了解底层的业务方可知如何传值,那整体来看增加沟通成本,对 ...

  4. Mybatis实战之自定义TypeHandler处理枚举

    在Mybatis中,处理枚举类的TypeHandler有两个: EnumTypeHandler: 用于保存枚举名 EnumOrdinalTypeHandler: 用于保存枚举的序号. 在实际项目中,以 ...

  5. mybatis自定义枚举转换类

    转载自:http://my.oschina.net/SEyanlei/blog/188919 mybatis提供了EnumTypeHandler和EnumOrdinalTypeHandler完成枚举类 ...

  6. mybatis 3的TypeHandler解析(null值的处理)

    最近,在测试迁移公司的交易客户端连接到自主研发的中间件时,调用DAO层时,发现有些参数并没有传递,而在mapper里面是通过parameterMap传递的,因为有些参数为null,这就导致了参数传递到 ...

  7. 解决mybatis使用枚举的转换

    解决mybatis使用枚举的转换 >>>>>>>>>>>>>>>>>>>>> ...

  8. mybatis入门系列三之类型转换器

    mybatis入门系列三之类型转换器 类型转换器介绍 mybatis作为一个ORM框架,要求java中的对象与数据库中的表记录应该对应 因此java类名-数据库表名,java类属性名-数据库表字段名, ...

  9. 两个案例轻松理解MyBatis中的TypeHandler!

    在做开发时,我们经常会遇到这样一些问题,比如我有一个Java中的Date数据类型,我想将之存到数据库的时候存成一个1970年至今的毫秒数,怎么实现?再比如我有一个User类,User类中有一个属性叫做 ...

随机推荐

  1. Android异步处理之AsyncTaskLoader简单使用

    简介 不管是在Android应用开发还是Android平台开发中,异步处理通常是最基本的coding要求.如果你还在主线程中写一些数据库,网络请求,读写本地文件等操作的话那说明你还不是一个合格的And ...

  2. Jmeter--BeanShell使用

    博客首页:http://www.cnblogs.com/fqfanqi/ (一)BeanShell简介 BeanShell是一个小型嵌入式Java源代码解释器,具有对象脚本语言特性,能够动态地执行标准 ...

  3. Java——BeanUtils基本用法

    为了操作JavaBean的属性,sun公司自己写了一套内省的api(在Java.beans.*)中,但是我们发现操作起来还是比较复杂的,所以apache公司就自己写了一套api替代了它,大大方便了开发 ...

  4. TADDConnetion组件,TADOQuery

    一.TADDConnetion 二.TADOQuery 1.RecNo:从1开始 当前记录行数;ADOQuery1.RecNo 选择后一行数据集内容:ADOQuery1.RecNo:=ADOQuery ...

  5. poj 1182 食物链 (带关系的并查集)

      食物链 Time Limit: 1000MS Memory Limit: 10000K Total Submissions: 44835 Accepted: 13069 Description 动 ...

  6. Java RTTI and Reflection

    Reference: Java编程思想 java 反射(Reflect) Java系列笔记(2) - Java RTTI和反射机制 Java Reflection in Action, 有空再补 -- ...

  7. JavaScript数据类型转换汇总

    ECMAScirpt中的数据类型:undefined.Null.Boolean.Number.String.Object 对一个值使用typeof操作符可能返回下列某个字符串: number(数字). ...

  8. Python中 and,or 的计算规则

    一.纯 and 和 or 语句 1. 在纯and语句中,如果每一个表达式都不是假的话,那么返回最后一个,因为需要一直匹配直到最后一个.如果有一个是假,那么返回假2. 在纯or语句中,只要有一个表达式不 ...

  9. python连接mongo

    连接mongodb数据库 用到pymongo模块 应该是这样来使用: , 'goods') 然后连接数据库层这么写 def getSpinfo(item,value,depart,comp): res ...

  10. Django之查询总结

    models.Book.objects.filter(**kwargs): querySet [obj1,obj2]models.Book.objects.filter(**kwargs).value ...