項目中使用枚舉類的好處這里不再贅述,在使用枚舉值時,通常需要根據值來獲取枚舉對象,下面介紹兩種實現方案:
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的初始化工作:

1 package com.example.myFirstProject.common; 2 3 import com.example.myFirstProject.service.EnumMessage; 4 import com.example.myFirstProject.util.PackageUtil; 5 import java.lang.reflect.Method; 6 import java.util.ArrayList; 7 import java.util.HashMap; 8 import java.util.List; 9 import java.util.Map; 10 11 public class Constant { 12 13 /** 14 * 枚舉類包名集合 15 */ 16 public static List<String> pathList = initPackagePathList(); 17 /** 18 * 枚舉接口類全路徑 19 */ 20 public final static String ENUM_MESSAGE_PATH = "com.example.myFirstProject.service.EnumMessage"; 21 22 /** 23 * 枚舉類對應的全路徑集合 24 */ 25 public static final List<String> ENUM_OBJECT_PATH = PackageUtil.getPackageClasses(pathList, true); 26 27 /** 28 * 存放單個枚舉對象 map常量定義 29 */ 30 private static Map<Object, EnumMessage> SINGLE_ENUM_MAP = null; 31 32 /** 33 * 所有枚舉對象的 map 34 */ 35 public static final Map<Class, Map<Object, EnumMessage>> ENUM_MAP = initialEnumMap(true); 36 37 private static List<String> initPackagePathList() { 38 List<String> list = new ArrayList<>(); 39 list.add("com.example.myFirstProject.enums"); 40 return list; 41 } 42 43 static { 44 System.out.println("類被加載時,先初始化各個靜態變量,再執行static塊。" + 45 "所以不能在這里執行pathList的add操作(\"com.example.myFirstProject.enums\")。"); 46 } 47 /** 48 * 加載所有枚舉對象數據 49 * 50 * @param isFouceCheck 是否強制校驗枚舉是否實現了EnumMessage接口,若為false則沒有實現接口的枚舉類也會被加載 51 */ 52 private static Map<Class, Map<Object, EnumMessage>> initialEnumMap(boolean isFouceCheck) { 53 Map<Class, Map<Object, EnumMessage>> ENUM_MAP = new HashMap<>(); 54 try { 55 for (String classname : ENUM_OBJECT_PATH) { 56 Class<?> cls = null; 57 cls = Class.forName(classname); 58 Class<?>[] iter = cls.getInterfaces(); 59 boolean flag = false; 60 if (isFouceCheck) { 61 for (Class cz : iter) { 62 if (cz.getName().equals(ENUM_MESSAGE_PATH)) { 63 flag = true; 64 break; 65 } 66 } 67 } 68 if (flag == isFouceCheck) { 69 SINGLE_ENUM_MAP = new HashMap<>(); 70 initialSingleEnumMap(cls); 71 ENUM_MAP.put(cls, SINGLE_ENUM_MAP); 72 } 73 74 } 75 } catch (Exception e) { 76 77 } 78 return ENUM_MAP; 79 } 80 81 /** 82 * 加載每個枚舉對象數據 83 */ 84 private static void initialSingleEnumMap(Class<?> cls) throws Exception { 85 Method method = cls.getMethod("values"); 86 EnumMessage inter[] = (EnumMessage[]) method.invoke(null, null); 87 for (EnumMessage enumMessage : inter) { 88 SINGLE_ENUM_MAP.put(enumMessage.getValue(), enumMessage); 89 } 90 } 91 92 93 }
PackageUtil工具類主要完成根據枚舉類所在包名獲取該package下所有class的全路徑名稱的工作:

1 package com.example.myFirstProject.util; 2 3 import java.io.File; 4 import java.util.ArrayList; 5 import java.util.List; 6 7 public class PackageUtil { 8 9 /** 10 * 返回包下所有的類 11 * 12 * @param packagePathList 包名全路徑集合 13 * @param classWithPath 返回全路徑開關 true 自動帶上包名 false 只返回類名 14 * @return List<String> 包下所有的類 15 */ 16 public static List<String> getPackageClasses(List<String> packagePathList, boolean classWithPath) { 17 List<String> result = new ArrayList<>(); 18 for(String packagePath : packagePathList) { 19 List<String> classNames = getClassName(packagePath); 20 String path = classWithPath ? packagePath + "." : ""; 21 for (String className : classNames) { 22 //className:com.example.myFirstProject.enums.SexEnum 23 result.add(path + className.substring(className.lastIndexOf(".") + 1)); 24 } 25 } 26 return result; 27 } 28 29 /** 30 * 獲取該報名全路徑下的所有class全路徑集合 31 * @param packageName 包名全路徑 32 * @return 33 */ 34 private static List<String> getClassName(String packageName) { 35 //根據報名獲取該package的系統路徑 36 String filePath = ClassLoader.getSystemResource("").getPath() + packageName.replace(".", "\\"); 37 // filePath: /D:/workspace-git/springbootlearning/target/classes/com\example\myFirstProject\enums 38 List<String> fileNames = getClassName(filePath, null); 39 return fileNames; 40 } 41 42 /** 43 * 獲取filePath文件夾下的所有class的全路徑集合 44 * @param filePath 45 * @param className 46 * @return 47 */ 48 private static List<String> getClassName(String filePath, List<String> className) { 49 List<String> myClassName = new ArrayList<>(); 50 File file = new File(filePath); 51 File[] childFiles = file.listFiles(); 52 for (File childFile : childFiles) { 53 if (childFile.isDirectory()) { 54 //遞歸獲取該文件夾下的子文件夾里的所有文件 55 myClassName.addAll(getClassName(childFile.getPath(), myClassName)); 56 } else { 57 String childFilePath = childFile.getPath(); 58 //childFilePath: D:\workspace-git\springbootlearning\target\classes\com\example\myFirstProject\enums\SexEnum.class 59 childFilePath = childFilePath.substring(childFilePath.indexOf("\\classes") + 9, childFilePath.lastIndexOf(".")); 60 childFilePath = childFilePath.replace("\\", "."); 61 myClassName.add(childFilePath); 62 } 63 } 64 65 return myClassName; 66 } 67 68 }
定義EnumUtil,提供根據值獲取枚舉對象的入口方法:

1 package com.example.myFirstProject.util; 2 3 import com.example.myFirstProject.common.Constant; 4 import com.example.myFirstProject.service.EnumMessage; 5 6 public class EnumUtil { 7 8 /** 9 * 獲取value返回枚舉對象 10 * @param value 11 * @param clazz 12 * */ 13 public static <T extends EnumMessage> T getEnumObject(Object value, Class<T> clazz){ 14 return (T) Constant.ENUM_MAP.get(clazz).get(value); 15 } 16 17 }
最后編寫測試語句:
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、調用類的靜態變量的時候;