引言

typeHandlers

阅读官方文档 typeHandlers 一节 {:target="_blank"}

MyBatis 在预处理语句(PreparedStatement)中设置一个参数时,Java对象将通过ps.setInt、ps.setString、ps.setTimeStamp等方法转换成数据库需要的数据

在从结果集(ResultSet)中取出一个值时,将使用rs.getInt、rs.getString、rs.getTimeStamp等方法将数据转换为Java对象

这两类操作通过 类型处理器 完成

  • 类型处理器均实现 TypeHandler<T> 接口,所有基本类型均有对应的类型处理器

  • Enum对应有两个类型处理器,分别为 EnumTypeHandler 、 EnumOrdinalTypeHandler

    若需将Enum字段映射为字符串,则使用 EnumTypeHandler 。 (默认使用)

    若需将Enum字段映射为int数值,则使用 EnumOrdinalTypeHandler

然而, EnumOrdinalTypeHandler 局限性非常明显,其映射的数据直接使用枚举值的 ordinal数值,因此与枚举值定义顺序紧耦合, 本文将解决此问题

一般合理实现方案(针对每一种特定值Enum定义专属TypeHandler)

假设存在一个产品类型枚举类型定义,其产品均对应有特定int值,实现如下

public enum ProductType {
AAA(100),
BBB(200),
CCC(300);
private int value;
private ProductType(int value) {
this.value = value;
}
public static ProductType fromValue(int value) {
for (ProductType productType : ProductType.values()) {
if (productType.value == value) {
return productType;
}
}
throw new IllegalArgumentException("Cannot create evalue from value: " + value + "!");
}
public int toValue() {
return value;
}
}

同时,需要定义对应的 ProductTypeHandler

public class ProductTypeHandler implements TypeHandler<ProductType> {
@Overridepublic void setParameter(PreparedStatement preparedStatement, int i, ProductType productType, JdbcType jdbcType)throws SQLException {
preparedStatement.setInt(i, productType.toValue());
}
@Overridepublic ProductType getResult(ResultSet resultSet, String s) throws SQLException {
return ProductType.fromValue(resultSet.getInt(s));
}
@Overridepublic ProductType getResult(ResultSet resultSet, int i) throws SQLException {
return ProductType.fromValue(resultSet.getInt(i));
}
@Overridepublic ProductType getResult(CallableStatement callableStatement, int i) throws SQLException {
return ProductType.fromValue(callableStatement.getInt(i));
}
}

最佳实践

同样以上方的 ProductType 举例,枚举类型的定义稍作修改

public enum ProductType implements ValuedEnum {
AAA(100),
BBB(200),
CCC(300);
private int value;
private ProductType(int value) {
this.value = value;
}
public int getValue() {
return value;
}
}

可以看出,该类型实现了接口 ValuedEnum ,同时去除了 fromValue 类方法, ValuedEnum 接口内容如下,仅声明一个 getValue 方法

public interface ValuedEnum {
int getValue();
}

不再需要为 ProductType 专属定义类型转换器,使用通用转换器 ValuedEnumTypeHanlder,使用方式同内置转换器 EnumOrdinalTypeHandler 完全一致。 实现如下

/**
* Created by sunlin05 on 2015/7/6.
* @author sunlin
*/
public class ValuedEnumTypeHandler<E extends Enum<E>> extends BaseTypeHandler<E> {
private Class<E> type;
private Map<Integer, E> map = new HashMap<>();
public ValuedEnumTypeHandler(Class<E> type) {
if (type == null) {
throw new IllegalArgumentException("Type argument cannot be null");
}
this.type = type;
E[] enums = type.getEnumConstants();
if (enums == null) {
throw new IllegalArgumentException(type.getSimpleName() + " does not represent an enum type.");
}
for (E e : enums) {
ValuedEnum valuedEnum = (ValuedEnum) e;
map.put(valuedEnum.getValue(), e);
}
}
@Override
public void setNonNullParameter(PreparedStatement ps, int i, E parameter, JdbcType jdbcType) throws SQLException {
ValuedEnum valuedEnum = (ValuedEnum) parameter;
ps.setInt(i, valuedEnum.getValue());
}
@Override
public E getNullableResult(ResultSet rs, String columnName) throws SQLException {
int i = rs.getInt(columnName);
if (rs.wasNull()) {
return null;
} else {
return getValuedEnum(i);
}
}
@Override
public E getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
int i = rs.getInt(columnIndex);
if (rs.wasNull()) {
return null;
} else {
return getValuedEnum(i);
}
}
@Override
public E getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {
int i = cs.getInt(columnIndex);
if (cs.wasNull()) {
return null;
} else {
return getValuedEnum(i);
}
}
private E getValuedEnum(int value) {
try {
return map.get(value);
} catch (Exception ex) {
throw new IllegalArgumentException(
"Cannot convert " + value + " to " + type.getSimpleName() + " by value.", ex);
}
}
}

该实现参考 EnumOrdinalTypeHandler 源码,核心逻辑见构造器,加注释说明如下

// 获取所有枚举值
E[] enums = type.getEnumConstants(); for (E e : enums) {
// 类型转换为ValuedEnum接口对象
ValuedEnum valuedEnum = (ValuedEnum) e; // 通过getValue()方法获取枚举值对应的Value int值,通过map记录映射关系
map.put(valuedEnum.getValue(), e);
}

Mybatis特殊值Enum类型转换器-ValuedEnumTypeHandler的更多相关文章

  1. [原创]Mybatis特殊值Enum类型转换器-ValuedEnumTypeHandler

    引言 typeHandlers 阅读官方文档 typeHandlers 一节{:target="_blank"} MyBatis 在预处理语句(PreparedStatement) ...

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

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

  3. (三)Mybatis类型转换器,接口传参类型,一对一,一对多查询resultMap配置

    Mybatis类型转换器 首先明白什么时候用到它,当数据库的字段类型和java字段类型无法默认匹配时候进行转换,比如现在数据库类型是INTEGER,而java当中类型是Boolean,true表示1, ...

  4. Mybatis入门——基础方式的增删该查、mapper动态代理方式的CRUD、类型转换器

    一.基础方式的增删该查: 1.mybatis约定:输入参数parameterType和输出参数resulrType在形式上只能有一个. 2.如果输入/输出参数:是简单类型(8个基本类型加String) ...

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

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

  6. Mybatis中使用自定义的类型处理器处理枚举enum类型

    知识点:在使用Mybatis的框架中,使用自定义的类型处理器处理枚举enum类型 应用:利用枚举类,处理字段有限,可以用状态码,代替的字段,本实例,给员工状态字段设置了一个枚举类 状态码,直接赋值给对 ...

  7. mybatis typeHandler类型转换器

    typeHandler类型转换器 在JDBC中,需要在PreparedStatement对象中设置那些已经预编译过的SQL语句的参数.执行SQL后,会通过ResultSet对象获取得到数据库的数据,而 ...

  8. mybatis 自定义类型转换器 (后台日期类型插入数据库)

    后台日期类型插入数据库 有以下几种发法: 1 调用数据库 日期字符串转日期函数 str_to_date("日期","yyyy-MM-dd HH:mm:ss") ...

  9. Struts2之自定义类型转换器

    Struts2自定义类型转换器分为局部类型转换器和全局类型转换器 (1)局部类型转换器 如果页面传来一个参数reg.action?birthday=2010-11-12到后台action,然后属性用d ...

随机推荐

  1. Revit API遍历全部风管,找到与风管相关的墙开洞

    涉及向量计算,求相交等相关技术. )                 {                     foreach (Face face in solid.Faces)          ...

  2. win8、server 2012 清除winsxs文件夹

    使用系统自带的文件清理工具 1.组合键win+x ,选择“命令提示符(管理员)” 复制“dism /online /Cleanup-Image /StartComponentCleanup” 到dos ...

  3. 前端使用AngularJS的$resource,后端ASP.NET Web API,实现分页、过滤

    在上一篇中实现了增删改查,本篇实现分页和过滤. 本系列包括: 1.前端使用AngularJS的$resource,后端ASP.NET Web API,实现增删改查2.前端使用AngularJS的$re ...

  4. Delphi下让窗口不显示在任务栏的另类方法

    刚才看到了这篇东西<使窗口不在任务栏上显示(利用ITaskbarList接口)>,作者用ITaskList接口实现了隐藏窗口在任务栏按钮的功能,想起我好多年以前做的程序也有这样的功能,但是 ...

  5. IIS7.5标识介绍

    应用程序池的标识是运行应用程序池的工作进程所使用的服务帐户名称.默认情况下,应用程序池以 Network Service 用户帐户运行,该帐户拥有低级别的用户权限.您可以将应用程序池配置为以 Wind ...

  6. 安装veloeclipse插件报错解决方案

    步骤: 1.把Eclipse安装目录下的artifacts.xml打开,搜索veloeclipse,把它相关的项删除: 2.Help 3. Install New Software 4.Work Wi ...

  7. 腾讯Bugly2015年移动应用质量大数据报告 原 荐

    在这份报告中,腾讯Bugly和腾讯优测会对2015年Android和iOS平台上的应用质量进行详细盘点,帮助你了解你的产品质量在行业中处于什么位置. 首先,让我们从整体上,回顾一下2015年度的应用和 ...

  8. linux 统计文件夹空间

     du -sh * | sort -nr 

  9. Orchard模块开发全接触5:深度改造前台第二部分

    在这一部分,我们继续完善我们的购物车,我们要做以下一些事情: 1:完成 shoppingcart.cshtml: 2:让用户可以更新数量及从购物车删除商品: 3:创建一个 widget,在上面可以看到 ...

  10. Asp.Net Core Web相对路径、绝对路径整理

    一.相对路径 1.关于Asp.Net Core中的相对路径主要包括两个部分:一.Web根目录,即当前网站的目录为基础:二.内容目录wwwroot文件夹,对于静态文件都放在这个目录. 2.获取控制器,A ...