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. python框架Scrapy中crawlSpider的使用

    一.创建Scrapy工程 #scrapy startproject 工程名 scrapy startproject demo3 二.进入工程目录,根据爬虫模板生成爬虫文件 #scrapy genspi ...

  2. 解决----------“win10,不能打字了,已禁用IME”

    登录Windows 10系统桌面后,右键点击左下角的开始菜单图标,然后在弹出菜单中选择“计算机管理”菜单项在打开的“计算机管理”窗口中,找到左侧系统工具下的“任务计划程序”菜单项点击任务计划程序前面的 ...

  3. cygwin简介,安装及卸载(体验UNIX & Linux环境)

    对于爱好者或初学者来说,为了体验UNIX & Linux环境,去安装虚拟机或双系统稍显麻烦,cygwin是一个很好的选择 具/原料   安装windows的电脑一台(可以联网) 法/步骤   ...

  4. Chrome 默认样式 (user agent stylesheetbody) 优先级变高的问题

    解决方法:只需要在页面的<HTML>标签前添加声明即可. <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional ...

  5. Java基础语法 - 面向对象 - this 关键字

    在Java语言中规定使用this关键字来代表本类对象的引用,this关键字被隐式地用于引用对象的成员变量和方法. this关键字引用的就是本类的一个对象,在局部变量或方法参数覆盖了成员变量时,就要添加 ...

  6. Java 面向对象编程介绍

    面向对象的概念 类与对象的关系 封装 面向对象 面向过程: 强调的是过程(动作) 面向对象: 强调的是对象(实体) 面向对象的特点 面向对象就是一种常见的思想,符合人们的思考习惯; 面向对象的出现,将 ...

  7. python多线程的两种写法

    1.一般多线程 import threading def func(arg): # 获取当前执行该函数的线程的对象 t = threading.current_thread() # 根据当前线程对象获 ...

  8. DRF(4) - 认证、权限组件

    一.引入 通过前面三节课的学习,我们已经详细了解了DRF提供的几个重要的工具,DRF充分利用了面向对象编程的思想,对Django的View类进行了继承,并封装了其as_view方法和dispatch方 ...

  9. Python中的不同进制的语法和转换

    不同进制的书写方式 八进制(Octal) 0o377 十六进制(Hex) 0xFF 二进制(Binary) 0b11111111 不同进制之间的转换 python提供了三个内置的函数,能够用来在不同进 ...

  10. unity3d相关资源

    http://pan.baidu.com/s/1kTG9DVD   GUI源码