MyBatis对于Java对象里的枚举类型处理

平时咱们写程序实体类内或多或少都会有枚举类型属性,方便嘛。但是mybatis里怎么处理他们的增删改查呢?

要求:

插入的时候,会用枚举的定义插入数据库,我们希望在数据库中看到的是数字或者其他东西;

查询的时候,数据库的值可以自动转换为我们对应的枚举值。

举例,我有一个这样的枚举类型:

Java Code复制内容到剪贴板

  1. package cn.com.shuyangyang.domain;
  2. public enum UserStatus {
  3. /** 无效*/
  4. DISABLED(0),
  5. /** 有效 */
  6. AVAILABLE(1);
  7. private int status;
  8. UserStatus(int status){
  9. this.status = status;
  10. }
  11. public int getStatus() {
  12. return status;
  13. }
  14. }

我们插入数据库中,数据库应该为我们保存0或者1

我们可以使用mybatis自带的枚举类型EnumOrdinalTypeHandler

举例如下:

Java Code复制内容到剪贴板

  1. <insert id="addUser" parameterType="User">
  2. INSERT INTO t_user(USER_ID,USER_NAME,LOGIN_NAME,
  3. LOGIN_PASSWORD,USER_STATUS,CREATE_TIME,UPDATE_TIME)
  4. VALUES(
  5. #{user_id},
  6. #{user_name},
  7. #{login_name},
  8. #{login_password},
  9. #{user_status, typeHandler=org.apache.ibatis.type.EnumOrdinalTypeHandler},
  10. #{create_time},
  11. #{update_time}
  12. )
  13. </insert>

我们的UserStatus插入的时候是UserStatus.AVAILABLE

执行完毕后,看结果:

看,是不是结果是我们期望的那样。

但是问题来了?

我们查询的时候报错了:Error querying database.  Cause: java.lang.IllegalArgumentException: No enum constant cn.com.shuyangyang.mybatis.UserStatus.1

我们可以自定义这样的一个EnumStatusHandler:

Java Code复制内容到剪贴板

  1. package cn.com.shuyangyang.mybatis;
  2. import java.sql.CallableStatement;
  3. import java.sql.PreparedStatement;
  4. import java.sql.ResultSet;
  5. import java.sql.SQLException;
  6. import org.apache.ibatis.type.BaseTypeHandler;
  7. import org.apache.ibatis.type.JdbcType;
  8. /**
  9. * Mybatis自定义转换类型
  10. * @author ShuYangYang
  11. * E-Mail:[email protected]
  12. * http://www.shuyangyang.com.cn
  13. * Date:2015年1月26日下午10:12:54
  14. *
  15. */
  16. public class EnumStatusHandler extends BaseTypeHandler<UserStatus> {
  17. private Class<UserStatus> type;
  18. private final UserStatus[] enums;
  19. /**
  20. * 设置配置文件设置的转换类以及枚举类内容,供其他方法更便捷高效的实现
  21. * @param type 配置文件中设置的转换类
  22. */
  23. public EnumStatusHandler(Class<UserStatus> type) {
  24. if (type == null)
  25. throw new IllegalArgumentException("Type argument cannot be null");
  26. this.type = type;
  27. this.enums = type.getEnumConstants();
  28. if (this.enums == null)
  29. throw new IllegalArgumentException(type.getSimpleName()
  30. + " does not represent an enum type.");
  31. }
  32. @Override
  33. public void setNonNullParameter(PreparedStatement ps, int i,
  34. UserStatus parameter, JdbcType jdbcType) throws SQLException {
  35. // baseTypeHandler已经帮我们做了parameter的null判断
  36. ps.setInt(i, parameter.getStatus());
  37. }
  38. @Override
  39. public UserStatus getNullableResult(ResultSet rs, String columnName)
  40. throws SQLException {
  41. // 根据数据库存储类型决定获取类型,本例子中数据库中存放INT类型
  42. int i = rs.getInt(columnName);
  43. if (rs.wasNull()) {
  44. return null;
  45. } else {
  46. // 根据数据库中的code值,定位EnumStatus子类
  47. return locateEnumStatus(i);
  48. }
  49. }
  50. @Override
  51. public UserStatus getNullableResult(ResultSet rs, int columnIndex)
  52. throws SQLException {
  53. // 根据数据库存储类型决定获取类型,本例子中数据库中存放INT类型
  54. int i = rs.getInt(columnIndex);
  55. if (rs.wasNull()) {
  56. return null;
  57. } else {
  58. // 根据数据库中的code值,定位EnumStatus子类
  59. return locateEnumStatus(i);
  60. }
  61. }
  62. @Override
  63. public UserStatus getNullableResult(CallableStatement cs, int columnIndex)
  64. throws SQLException {
  65. // 根据数据库存储类型决定获取类型,本例子中数据库中存放INT类型
  66. int i = cs.getInt(columnIndex);
  67. if (cs.wasNull()) {
  68. return null;
  69. } else {
  70. // 根据数据库中的code值,定位EnumStatus子类
  71. return locateEnumStatus(i);
  72. }
  73. }
  74. /**
  75. * 枚举类型转换,由于构造函数获取了枚举的子类enums,让遍历更加高效快捷
  76. * @param code 数据库中存储的自定义code属性
  77. * @return code对应的枚举类
  78. */
  79. private UserStatus locateEnumStatus(int code) {
  80. for(UserStatus status : enums) {
  81. if(status.getStatus()==(Integer.valueOf(code))) {
  82. return status;
  83. }
  84. }
  85. throw new IllegalArgumentException("未知的枚举类型:" + code + ",请核对" + type.getSimpleName());
  86. }
  87. }

mapper.xml里这样配置:

XML/HTML Code复制内容到剪贴板

  1. <resultMap type="User" id="userResult">
  2. ……省略其他属性配置
  3. <result column="USER_STATUS" property="user_status" typeHandler="cn.com.shuyangyang.mybatis.EnumStatusHandler"/>
  4. ……省略其他属性配置
  5. </resultMap>

现在来看看结果:

[User [user_id=782cba19-559f-41c3-a1b0-25fcac4cf118, user_name=系统管理员, login_name=admin, login_password=admin,user_status=AVAILABLE, create_time=Mon Jan 26 21:17:09 CST 2015, update_time=Mon Jan 26 21:17:09 CST 2015]]

完美结果。

来源: http://www.shuyangyang.com.cn/jishuliangongfang/Javabiancheng/2015-01-26/229.html

来自为知笔记(Wiz)

时间: 2024-08-08 23:34:53

MyBatis对于Java对象里的枚举类型处理的相关文章

Java基础教程:枚举类型

Java基础教程:枚举类型 枚举类型 枚举是将一具有类似特性的值归纳在一起的方法.比如,我们可以将周一到周日设计为一个枚举类型.彩虹的七种颜色设计为一个枚举类型. 常量实现枚举 我们通过定义常量的方式来实现,如下: Public static class RainbowColor { // 红橙黄绿青蓝紫七种颜色的常量定义 public static final int RED = 0; public static final int ORANGE = 1; public static fina

Java魔法堂:枚举类型详解

一.前言 Java的枚举类型相对C#来说具有更灵活可配置性,Java的枚举类型可以携带更多的信息. // C# enum MyColor{ RED = 0, BLUE = 1 } Console.Write(MyColor.RED); // Java enum MyColor{ RED("Hot", 4), BLUE("SAD",8); private String mood; public String getMood{ return mood; } privat

Java基础教程(15)--枚举类型

??枚举类型定义了一个枚举值的列表,每个值是一个标识符.例如,下面的语句声明了一个枚举类型,用来表示星期的可能情况: public enum Day { SUNDAY, MONDAY, TUESDAY, WEDNESDAY, THURSDAY, FRIDAY, SATURDAY } ??实际上,这个声明定义的类型是一个类,它有7个实例.因此,在比较两个枚举类型的值时,永远不需要调用equals,直接使用"=="就可以了. ??枚举类型的值类似于一个常量,因此,按照惯例拼写都使用大写字母

Oracle 中Java 对象与PL/SQL类型的映射及使用

一.Jpublisher概述 Oracle JPublisher是一个用于在Java和SQL之间转换和交换结构化对象数据的工具,它还可以访问其他的数据库资源,如PL/SQL包和Java存储 过程.它生成Java类来表示数据库实体,如SQL对象和操作.PL/SQL包和过程以及服务器端Java类.你可以通过JDBC在你的Java客户端. servlet.JavaServer Pages (jsp).Enterprise JavaBeans (EJB)和Web服务中使用这些生成的类. JPublish

Java练习 SDUT-1959_简单枚举类型——植物与颜色

简单枚举类型--植物与颜色 Time Limit: 1000 ms Memory Limit: 65536 KiB Problem Description 请定义具有red, orange, yellow, green, blue, violet六种颜色的枚举类型color,根据输入的颜色名称,输出以下六种植物花朵的颜色: Rose(red), Poppies(orange), Sunflower(yellow), Grass(green), Bluebells(blue), Violets(v

枚举器对象与可枚举类型

1.IEnumerator与IEnumerable接口 IEnumerator接口:实现此接口的为枚举器类型,此接口包含3个方法Current.MoveNext.Reset. IEnumerable接口:实现此接口只需实现一个方法GetEnumerator得到一个枚举器. class ColorEnumerator : IEnumerator //实现枚举器接口:Current,MoveNext,Reset共3种方法 { string[] _colors; int _position = -1;

java入参校验——枚举类型

一般来讲,对一些post请求,我们可以使用@valid+javax.validation.constraints包下注解的方式,优雅把参数验证放在control层的requst中,举例说明如下图: 但是此时,如果我们知道可以修改的状态只能为枚举类中存在的值,并没有已有的注解可以用,我们可以采用以下代码来实现此功能: 首先我们自己实现一个验证注解,具体代码如下 @Target({ElementType.FIELD, ElementType.METHOD, ElementType.ANNOTATIO

深入理解Java枚举类型(enum)

[版权申明]未经博主同意,谢绝转载!(请尊重原创,博主保留追究权) http://blog.csdn.net/javazejian/article/details/71333103 出自[zejian的博客] 关联文章: 理解Java位运算 深入理解Java类型信息(Class对象)与反射机制 本篇主要是深入对Java中枚举类型进行分析,主要内容如下: 理解枚举类型 枚举的定义 枚举实现原理 枚举的常见方法 Enum抽象类常见方法 编译器生成的Values方法与ValueOf方法 枚举与Clas

深入Java 1.5枚举类型的内部实现原理

Java是一种面向对象的高级编程语言.它的出众之处就在于它的简洁.一个程序员所要做的就是创建类(Create Class)以及定义接口(Define Interface),如此而已.当然,这种简洁和优美是有代价的,比如失去了Enum这种广泛使用的数据类型就是一个不小的损失.在Java 1.5以前,程序员们不得不通过一些变通的方法来间接的解决这一问题.比如说,被普遍使用的整数枚举替代法和类型安全类替代法(Type safe Enum).在正式讨论Java 1.5的枚举类型之前,让我们先简单回顾一下