[原创]Mybatis特殊值Enum类型转换器-ValuedEnumTypeHandler
引言
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> {
    @Override
    public void setParameter(PreparedStatement preparedStatement, int i, ProductType productType, JdbcType jdbcType)
            throws SQLException {
        preparedStatement.setInt(i, productType.toValue());
    }
    @Override
    public ProductType getResult(ResultSet resultSet, String s) throws SQLException {
        return ProductType.fromValue(resultSet.getInt(s));
    }
    @Override
    public ProductType getResult(ResultSet resultSet, int i) throws SQLException {
        return ProductType.fromValue(resultSet.getInt(i));
    }
    @Override
    public 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的更多相关文章
- Mybatis特殊值Enum类型转换器-ValuedEnumTypeHandler
		引言 typeHandlers 阅读官方文档 typeHandlers 一节 {:target="_blank"} MyBatis 在预处理语句(PreparedStatement ... 
- mybatis类型转换器 - 自定义全局转换enum
		在数据模型.接口参数等场景部分属性参数为一些常量值,比如性别:男.女.若是定义成int或String类型,于是类型本身的范围太宽,要求使用者需要了解底层的业务方可知如何传值,那整体来看增加沟通成本,对 ... 
- (三)Mybatis类型转换器,接口传参类型,一对一,一对多查询resultMap配置
		Mybatis类型转换器 首先明白什么时候用到它,当数据库的字段类型和java字段类型无法默认匹配时候进行转换,比如现在数据库类型是INTEGER,而java当中类型是Boolean,true表示1, ... 
- Mybatis入门——基础方式的增删该查、mapper动态代理方式的CRUD、类型转换器
		一.基础方式的增删该查: 1.mybatis约定:输入参数parameterType和输出参数resulrType在形式上只能有一个. 2.如果输入/输出参数:是简单类型(8个基本类型加String) ... 
- [原创]java WEB学习笔记67:Struts2 学习之路-- 类型转换概述, 类型转换错误修改,如何自定义类型转换器
		本博客的目的:①总结自己的学习过程,相当于学习笔记 ②将自己的经验分享给大家,相互学习,互相交流,不可商用 内容难免出现问题,欢迎指正,交流,探讨,可以留言,也可以通过以下方式联系. 本人互联网技术爱 ... 
- (转+原创)java的枚举类型Enum解释
		原文:http://www.cnblogs.com/mxmbk/articles/5091999.html 下文中还添加了个人的一些补充和理解. 在Java SE5之前,我们要使用枚举类型时,通常会使 ... 
- mybatis入门系列三之类型转换器
		mybatis入门系列三之类型转换器 类型转换器介绍 mybatis作为一个ORM框架,要求java中的对象与数据库中的表记录应该对应 因此java类名-数据库表名,java类属性名-数据库表字段名, ... 
- Mybatis中使用自定义的类型处理器处理枚举enum类型
		知识点:在使用Mybatis的框架中,使用自定义的类型处理器处理枚举enum类型 应用:利用枚举类,处理字段有限,可以用状态码,代替的字段,本实例,给员工状态字段设置了一个枚举类 状态码,直接赋值给对 ... 
- mybatis typeHandler类型转换器
		typeHandler类型转换器 在JDBC中,需要在PreparedStatement对象中设置那些已经预编译过的SQL语句的参数.执行SQL后,会通过ResultSet对象获取得到数据库的数据,而 ... 
随机推荐
- 【Hive】数据去重
			实现数据去重有两种方式 :distinct 和 group by 1.distinct消除重复行 distinct支持单列.多列的去重方式. 单列去重的方式简明易懂,即相同值只保留1个. 多列的去重则 ... 
- mysql注入介绍
			0. SQL注入常用的尝试语句: or 1=1--+ 'or 1=1--+ "or 1=1--+ )or 1=1--+ ')or 1=1--+ ") or 1=1--+ " ... 
- ViewGroup的事件分发机制
			我们用手指去触摸Android手机屏幕,就会产生一个触摸事件,但是这个触摸事件在底层是怎么分发的呢?这个我还真不知道,这里涉及到操作硬件 (手机屏幕)方面的知识,也就是Linux内核方面的知识,我也没 ... 
- Audio/Movie/Image
			Audio 1. 引入AVFoundation 库,此库用于处理音频的播放. > 使用AVAudioPlayer 播放音频,此类只能播放本地音频文件.对于流媒体(边下边播)的播放使用第三方框架实 ... 
- Ganymed实现基本的自动化部署API
			Ganymed SSH-2 for Java是一个纯Java实现的SHH2库,官网为http://www.ganymed.ethz.ch/ssh2/,最新的更新时间为2006年10月,在用之前,请仔细 ... 
- php之opcodes
			opcode是一种php脚本编译之后的语言. 例如: <?php echo "Hello World"; $a = 1 + 1; echo $a; ?> php执行这段 ... 
- HihoCoder1621 : 超市规划(四边形DP优化)()
			超市规划 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi居住的城市的中轴线恰好是一条马路.沿着这条马路一共坐落有N个居民小区,其中第i个小区距离马路一端的距离是A ... 
- Windows下运行Hadoop
			Windows下运行Hadoop,通常有两种方式:一种是用VM方式安装一个Linux操作系统,这样基本可以实现全Linux环境的Hadoop运行:另一种是通过Cygwin模拟Linux环境.后者的好处 ... 
- Python学习-数据运算
			在Python中有丰富的算术运算,这使得Python在科学计算领域有着很高的地位,Python可以提供包括四则运算在内的各种算术运算. a = 10 b = 20 print(a-b) #-10 pr ... 
- AngularJS方法  —— angular.bootstrap
			描述: 此方法用于手动加载angularjs模板 (官方翻译:注意基于端到端的测试不能使用此功能来引导手动加载,他们必须使用ngapp. angularjs会检测这个模板是否被浏览器加载或者加载多次并 ... 
