引言

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. WinForm MDIParent如何防止重复打开

    DI,全称是多文档界面(Multiple Document Interface),主要应用于基于图形用户界面的系统中.其目的是同时打开和显示多个文档,便于参考和编辑资料. 下面是一个WinForm M ...

  2. In House打包流程

    在一个app历经数周持续开发和多个版本快速内部迭代之后,当我们需要把这个版本发布到我们实际应用场景中,面对我们真实用户去say hi时,如果自身产品在发布(内测版本)之前确实找到一些潜在切相对稳定的种 ...

  3. 在Visual Studio中使用活动图描述业务流程

    当希望描述某个流程的时候,用活动图表示. 在项目中添加一个名称为"Shopping"的文件夹. 把"Orders Model"这个UML类图拖放到Shoppin ...

  4. Quartz.NET简介

    Quartz是OpenSymphony开源组织在Job scheduling领域又一个开源项目,它可以与J2EE与J2SE应用程序相结合也可以单独使用.Quartz可以用来创建简单或为运行十个,百个, ...

  5. ios之快速枚举

    for(UIView * subView in self.view.subviews) { if([subView isKindOfClass:[XYZSeniorQueryView class]]) ...

  6. Java锁的设计

    1.自旋锁 自旋锁是采用让当前线程不停地的在循环体内执行实现的,当循环的条件被其他线程改变时 才能进入临界区.如下 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 public ...

  7. SQLite中的事务操作

    关于SQLite事务可以解决一些问题,比如你要插入两个数据,可以将两个数据作为同一个事务进行插入,这样如果第二个数据错误了,便自动执行回滚操作,第一个数据也不会插入成功,保证了数据的同步! 一.实际的 ...

  8. FreeMarker:Java模板引擎

    本资源由 伯乐在线 - 黄余粮 整理 FreeMarker 是一个模板引擎,一个基于模板生成文本输出的通用工具,使用纯Java编写.FreeMarker被设计用来生成HTML Web页面,特别是基于M ...

  9. 带标签的 break 和 continue (Java)

    2002年12月发于 CCW-I2006年9月搬至 CSDN,略作修改 Java 中有 goto 关键字,但这个关键字没有任何作用,换句话说,我们不能使用 goto 来进行跳转到某行.实际上,结构化程 ...

  10. 基于图的图像分割(Graph-Based Image Segmentation)

    一.介绍 基于图的图像分割(Graph-Based Image Segmentation),论文<Efficient Graph-Based Image Segmentation>,P. ...