项目中使用枚举类的好处这里不再赘述,在使用枚举值时,通常需要根据值来获取枚举对象,下面介绍两种实现方案:

1.在枚举类中定义方法实现

  首先给出如下性别枚举类:  

public enum SexEnum {
MAN("M", "男"),
WOMAN("F", "女"); private String code;
private String desc; SexEnum(String code, String desc) {
this.code = code;
this.desc = desc;
} public String getCode() {
return code;
} public void setCode(String code) {
this.code = code;
} public String getDesc() {
return desc;
} public void setDesc(String desc) {
this.desc = desc;
} }

  现在需要根据code的值获取枚举对象,简单直接的办法是在该枚举类中定义如下方法:

  public static SexEnum getSexEnumByCode(String code){
for(SexEnum sexEnum : SexEnum.values()){
if(StringUtils.equals(code, sexEnum.getCode())){
return sexEnum;
}
}
return null;
}

  以这种方案实现时,需要在每个枚举类中都定义类似上述结构的方法。当项目中的枚举类较多时,显得代码冗余。

2.利用反射实现

  首先介绍本方案的实现方式,再来介绍具体代码实现:

  1).定义一个EnumMessage接口,然后每个枚举类实现此接口;

  2).定义常量保存枚举类所在包名,以及接口全路径;

  3).在程序启动时,读取枚举类所在包下的所有枚举类的File文件,在从file文件信息中获取每个枚举类的全路径类名集合A;

  4).遍历A集合,利用反射获取每个类的class对象,再判断该类是否实现了EnumMessage接口;

  5).对于实现了EnumMessage接口的枚举类,遍历该枚举类的所有对象,保存Map<Object, EnumMessage>的集合映射;

  6).对枚举类保存Map<Class, Map<Object, EnumMessage>>的映射集合。

  至此完成了启动的初始化工作。下面给出上述过程的代码实现:

  定义接口EnumMessage:

package com.example.myFirstProject.service;

public interface EnumMessage {
Object getValue();
}

  枚举类SexEnum实现此接口: 

package com.example.myFirstProject.enums;

import com.example.myFirstProject.service.EnumMessage;
import org.apache.commons.lang3.StringUtils; public enum SexEnum implements EnumMessage {
MAN("M", "男"),
WOMAN("F", "女"); private String code;
private String desc; SexEnum(String code, String desc) {
this.code = code;
this.desc = desc;
} public String getCode() {
return code;
} public void setCode(String code) {
this.code = code;
} public String getDesc() {
return desc;
} public void setDesc(String desc) {
this.desc = desc;
} @Override
public Object getValue() {
//此处需要根据枚举对象的哪个属性返回枚举对象,就return该属性
return code;
} }

  Constant类定义了常量保存枚举类所在包名和接口全路径,以及Map的初始化工作:

 package com.example.myFirstProject.common;

 import com.example.myFirstProject.service.EnumMessage;
import com.example.myFirstProject.util.PackageUtil;
import java.lang.reflect.Method;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map; public class Constant { /**
* 枚举类包名集合
*/
public static List<String> pathList = initPackagePathList();
/**
* 枚举接口类全路径
*/
public final static String ENUM_MESSAGE_PATH = "com.example.myFirstProject.service.EnumMessage"; /**
* 枚举类对应的全路径集合
*/
public static final List<String> ENUM_OBJECT_PATH = PackageUtil.getPackageClasses(pathList, true); /**
* 存放单个枚举对象 map常量定义
*/
private static Map<Object, EnumMessage> SINGLE_ENUM_MAP = null; /**
* 所有枚举对象的 map
*/
public static final Map<Class, Map<Object, EnumMessage>> ENUM_MAP = initialEnumMap(true); private static List<String> initPackagePathList() {
List<String> list = new ArrayList<>();
list.add("com.example.myFirstProject.enums");
return list;
} static {
System.out.println("类被加载时,先初始化各个静态变量,再执行static块。" +
"所以不能在这里执行pathList的add操作(\"com.example.myFirstProject.enums\")。");
}
/**
* 加载所有枚举对象数据
*
* @param isFouceCheck 是否强制校验枚举是否实现了EnumMessage接口,若为false则没有实现接口的枚举类也会被加载
*/
private static Map<Class, Map<Object, EnumMessage>> initialEnumMap(boolean isFouceCheck) {
Map<Class, Map<Object, EnumMessage>> ENUM_MAP = new HashMap<>();
try {
for (String classname : ENUM_OBJECT_PATH) {
Class<?> cls = null;
cls = Class.forName(classname);
Class<?>[] iter = cls.getInterfaces();
boolean flag = false;
if (isFouceCheck) {
for (Class cz : iter) {
if (cz.getName().equals(ENUM_MESSAGE_PATH)) {
flag = true;
break;
}
}
}
if (flag == isFouceCheck) {
SINGLE_ENUM_MAP = new HashMap<>();
initialSingleEnumMap(cls);
ENUM_MAP.put(cls, SINGLE_ENUM_MAP);
} }
} catch (Exception e) { }
return ENUM_MAP;
} /**
* 加载每个枚举对象数据
*/
private static void initialSingleEnumMap(Class<?> cls) throws Exception {
Method method = cls.getMethod("values");
EnumMessage inter[] = (EnumMessage[]) method.invoke(null, null);
for (EnumMessage enumMessage : inter) {
SINGLE_ENUM_MAP.put(enumMessage.getValue(), enumMessage);
}
} }

  PackageUtil工具类主要完成根据枚举类所在包名获取该package下所有class的全路径名称的工作:

 package com.example.myFirstProject.util;

 import java.io.File;
import java.util.ArrayList;
import java.util.List; public class PackageUtil { /**
* 返回包下所有的类
*
* @param packagePathList 包名全路径集合
* @param classWithPath 返回全路径开关 true 自动带上包名 false 只返回类名
* @return List<String> 包下所有的类
*/
public static List<String> getPackageClasses(List<String> packagePathList, boolean classWithPath) {
List<String> result = new ArrayList<>();
for(String packagePath : packagePathList) {
List<String> classNames = getClassName(packagePath);
String path = classWithPath ? packagePath + "." : "";
for (String className : classNames) {
//className:com.example.myFirstProject.enums.SexEnum
result.add(path + className.substring(className.lastIndexOf(".") + 1));
}
}
return result;
} /**
* 获取该报名全路径下的所有class全路径集合
* @param packageName 包名全路径
* @return
*/
private static List<String> getClassName(String packageName) {
//根据报名获取该package的系统路径
String filePath = ClassLoader.getSystemResource("").getPath() + packageName.replace(".", "\\");
// filePath: /D:/workspace-git/springbootlearning/target/classes/com\example\myFirstProject\enums
List<String> fileNames = getClassName(filePath, null);
return fileNames;
} /**
* 获取filePath文件夹下的所有class的全路径集合
* @param filePath
* @param className
* @return
*/
private static List<String> getClassName(String filePath, List<String> className) {
List<String> myClassName = new ArrayList<>();
File file = new File(filePath);
File[] childFiles = file.listFiles();
for (File childFile : childFiles) {
if (childFile.isDirectory()) {
//递归获取该文件夹下的子文件夹里的所有文件
myClassName.addAll(getClassName(childFile.getPath(), myClassName));
} else {
String childFilePath = childFile.getPath();
//childFilePath: D:\workspace-git\springbootlearning\target\classes\com\example\myFirstProject\enums\SexEnum.class
childFilePath = childFilePath.substring(childFilePath.indexOf("\\classes") + 9, childFilePath.lastIndexOf("."));
childFilePath = childFilePath.replace("\\", ".");
myClassName.add(childFilePath);
}
} return myClassName;
} }

  定义EnumUtil,提供根据值获取枚举对象的入口方法: 

 package com.example.myFirstProject.util;

 import com.example.myFirstProject.common.Constant;
import com.example.myFirstProject.service.EnumMessage; public class EnumUtil { /**
* 获取value返回枚举对象
* @param value
* @param clazz
* */
public static <T extends EnumMessage> T getEnumObject(Object value, Class<T> clazz){
return (T) Constant.ENUM_MAP.get(clazz).get(value);
} }

  最后编写测试语句:

  System.out.println(EnumUtil.getEnumObject("M", SexEnum.class));  //MAN

  至此该方案实现了根据枚举对象的值"M"获取枚举类对象"MAN"。

  注意:关于static变量的加载时机:

  当在EnumUtil中调用Constant的静态变量ENUM_MAP时,Constant类被加载,Conatant类中的pathList,ENUM_OBJECT_PATH,ENUM_MAP被按顺序加载,即先执行了Conatant的initPackagePathList()方法,再执行了PackageUtil的getPackageClasses(pathList, true)方法

最后在 public static final Map<Class, Map<Object, EnumMessage>> ENUM_MAP = initialEnumMap(true)被调用时,ENUM_OBJECT_PATH已经有值。

  附:类被加载的时机:  

  1、用Class.forName()显示加载的时候;

  2、实例化一个类的时候;

  3、调用类的静态方法的时候;

  4、调用类的静态变量的时候;

第三章 EnumUtil根据值获取枚举对象的更多相关文章

  1. java 根据值获取枚举对象

    关键方法: /** * 值映射为枚举 * * @param enumClass 枚举类 * @param value 枚举值 * @param method 取值方法 * @param <E&g ...

  2. Javascript权威指南——第二章词法结构,第三章类型、值和变量,第四章表达式和运算符,第五章语句

    第二章 词法结构 一.HTML并不区分大小写(尽管XHTML区分大小写),而javascript区分大小写:在HTML中,这些标签和属性名可以使用大写也可以使用小写,而在javascript中必须小写 ...

  3. 《JS权威指南学习总结--第三章类型、值和变量》

    第三章 类型.值和变量 内容要点 一.数据类型 1.在编程语言中,能够表示并操作的值的类型称做数据类型 2.JS的数据类型分为两类: 原始类型:数字.字符串和布尔值 对象类型 3.JS中有两个特殊的原 ...

  4. 根据值获取枚举类对象工具类EnumUtils

    枚举类 public enum Sex { man("M","男"),woman("W","女"); private S ...

  5. java反射之获取枚举对象

    项目中导入大量枚举对象,用来定义常量.随着带来一个问题,就是每个枚举类都需要通过key来获取对应枚举的需求. public enum ExamType { CRAFT(1, "草稿" ...

  6. 【笔记】javascript权威指南-第三章-类型,值和变量

    javascript中的原始类型和对象类型(基本类型和引用类型) //本书是指:javascript权威指南    //以下内容摘记时间为:2013.7.27   计算机程序运行时需要对值(value ...

  7. ES6 常用总结——第三章(数组、函数、对象的扩展)

    1.1. Array.from() Array.from方法用于将两类对象转为真正的数组:类似数组的对象(array-like object)和可遍历(iterable)的对象(包括ES6新增的数据结 ...

  8. js通过值获取数组对象对应下标

    var nn = [ { a: 'ss' },{ a: 'aa' },{ a : '11'},{ a: '33' },{ a: '88' } ] 我要怎么获取 a = 33的下标 var index ...

  9. H5_0017:通过元素自定义属性值获取元素对象,并获取属性值

            // 通过元素的属性值查找对象         // document.querySelectorAll("[data]").forEach(function(e) ...

随机推荐

  1. How to acquire an Android phone with locked bootloader?

    As we know that some devices come with locked bootloaders like Sony, HUAWEI, hTC...If you try to unl ...

  2. Broadcom以太网交换芯片培训

      目录 1.交换芯片架构....................................................................................... ...

  3. 交换芯片收发包的 DMA 实现原理

    交换芯片支持:报文.计数.表项3种DMA类型,其中报文DMA包括系统从芯片到接收报文或发送报文到交换芯片,计数DMA用来从片上获取统计计数,表项DMA功能分为SLAM DMA(系统内存DMA到片上交换 ...

  4. Windows出现带空格文件名无法删除

    下午同事的电脑上突然出现一个文件夹,怎么也删除不了,也无法重命名. 直接删除文件夹提示:无法读取源文件或磁盘. 然后查相关进程,未发现异常.重启进安全模式下删除,一样的结果,提示:无法读取源文件或磁盘 ...

  5. js 小工具-- 按长度截取字符串

    function cutstr(str,len){ var temp; var icount = 0; var stren = ""; //这段正则表示匹配所有汉字以及全角字符 v ...

  6. Programming Entity Framework 翻译(1)-目录

    1. Introducing the ADO.NET Entity Framework ado.net entity framework 介绍 1 The Entity Relationship Mo ...

  7. [转]逻辑斯蒂回归 via python

    # -*- coding:UTF-8 -*-import numpydef loadDataSet(): return dataMat,labelMat def sigmoid(inX): retur ...

  8. java的继承和重写

    继承是面向对象最显著的一个特性.继承是从已有的类中派生出新的类,新的类能吸收已有类的数据属性和行为,并能扩展新的能力.[1]  Java继承是使用已存在的类的定义作为基础建立新类的技术,新类的定义可以 ...

  9. Android之使用Android-query框架进行开发(一)(转载)

    开发Android使用Android-query框架能够快速的,比传统开发android所要编写的代码要少得很多,容易阅读等优势. 下载文档及其例子和包的地址:http://code.google.c ...

  10. 微信支付开发-当前页面的URL未注册

    微信支付的开发设置要求设置微信支付可以发起的目录. 当出现“当前页面的URL未注册”的错误时,检查你的授权目录(一般是未填写或填写有误),注意要二级或者三级目录完全匹配.