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

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. MS - 2 - 设计包含 min 函数的栈

    定义栈的数据结构,要求添加一个 min 函数,能够得到栈的最小元素. 要求函数 min.push 以及 pop 的时间复杂度都是 O(1). template<typename T> st ...

  2. vbox 不识别u盘的问题

    usb设备 ->启用usb设备 ->启用usb2.0(ehci)控制器 添加usb筛选器 给筛选器起个名字

  3. 拒绝访问 temp 目录。用来运行 XmlSerializer 的标识“IIS APPPOOL\UGAS”没有访问 temp 目录的足够权限

    在部署IIS时候会出现下图错误,拒绝访问 temp 目录.用来运行 XmlSerializer 的标识“IIS APPPOOL\UGAS”没有访问 temp 目录的足够权限 解决方法: 在IIS信息管 ...

  4. Sublim Text3快捷键大全

    Ctrl+Shift+P:打开命令面板Ctrl+P:搜索项目中的文件Ctrl+G:跳转到第几行Ctrl+W:关闭当前打开文件Ctrl+Shift+W:关闭所有打开文件Ctrl+Shift+V:粘贴并格 ...

  5. performSelector和respondsToSelector用法

    一.performSelector调用和直接调用区别 下面两段代码都在主线程中运行,我们在看别人代码时会发现有时会直接调用,有时会利用performSelector调用,今天看到有人在问这个问题,我便 ...

  6. 使用noConflict重命名jQuery对象

    大多数JavaScript框架都使用$符号作为缩写,当在同一个页面使用多个JS框架时,页面很容易发生冲突.幸运的是有一个简单的方法.noConflict()函数交回$的控制权并允许你设置成自己的变量名 ...

  7. 转: python如何安装pip和easy_installer工具

    原文地址: http://blog.chinaunix.net/uid-12014716-id-3859827.html 1.在以下地址下载最新的PIP安装文件:http://pypi.python. ...

  8. invoke

    在用.NET Framework框架的WinForm构建GUI程序界面时,如果要在控件的事件响应函数中改变控件的状态,例如:某个按钮上的文本原先叫“打开”,单击之后按钮上的文本显示“关闭”,初学者往往 ...

  9. CheckedComboBoxEdit 重置初始化值的方法

    CheckedComboBoxEdit ccbgb; ccbgb绑定方式ccbgb .Properties .DataSource =数据集; ccbgb.Properties.DisplayMemb ...

  10. [原创]Lodop打印, 以及Lodop引用css文件控制打印样式的问题.

    最近在做Lodop打印功能: 思路是:  用MasterPage搭个打印页面的框架, 然后在具体的页面中填入数据, 打印的样式由母版页和CSS来控制. 困扰了一天的问题是:  在打印的JS文件中, 引 ...