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. 并发编程6 锁&进程&队列

    1.进程的其他方法 2.验证进程空间隔离和守护进程 3.孤儿进程和僵尸进程 4.锁 for循环加join数据共享 5.进程队列的简单应用 6.通过队列实现进程间的通信 7.生产者消费者模型及Queue ...

  2. [科普]什么是SysWow64

    Wow!什么是Wow64 今天有个同事,被SysWow64搞晕了.这里简单介绍一下. 64位的Windows并不是简单地把所有东西都编译成64位就万事大吉的.关于64位的CPU应该做成什么样子,Int ...

  3. python迭代器、生成器、yield和xrange

    https://blog.csdn.net/u010138758/article/details/56291013

  4. css的常用知识点

    一.css的引入方式 1.行内引入 直接在标签中定义样式 <p style="background-color: red">test</p> 2.嵌入式引入 ...

  5. 从es中拉取全部数据/大量数据 使用scroll+scan避免深分页

    es一次请求默认返回的数据条数是10条,可以通过设置size参数来控制返回数据的条数: 如果要返回很多数据,可以把size设置的很大,不过elastic search默认size最大不能超过1万. 那 ...

  6. 创建使用pycharm virtualenv

    创建使用pycharm virtualenv 在python的世界里,真该感谢有PyCharm,pip,virtualenv 这些好东东,为python程序员提供了极大的方便. virtualenv ...

  7. 编译安装cmake3

    编译安装cmake3 ubuntu 14 的系统默认安装的是cmake2.7,apt-get upgrade之后也还是cmake2.7,而很多软件如今需要3及以上的版本来进行cmake编译(如caff ...

  8. 面试常见的selenium问题

    1.如何切换iframe 问题:如果你在一个default content中查找一个在iframe中的元素,那肯定是找不到的.反之你在一个iframe中查找另一个iframe元素或default co ...

  9. Python 一键同步windows和linux数据(基于pscp.exe)

    outline 项目中需要把 windows server 上的数据同步到 linux server,方法很多,这里记录下自己采用的一种比较简单的方法. 准备工作 首先确保你 windows serv ...

  10. python创建进程的两种方式

    1.方式1 import time import multiprocessing def task(arg): time.sleep(2) print(arg) def run(): # 进程1 p1 ...