第三章 EnumUtil根據值獲取枚舉對象


  

  項目中使用枚舉類的好處這里不再贅述,在使用枚舉值時,通常需要根據值來獲取枚舉對象,下面介紹兩種實現方案:

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;
  }

}
View Code

  現在需要根據code的值獲取枚舉對象,簡單直接的辦法是在該枚舉類中定義如下方法:

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

  以這種方案實現時,需要在每個枚舉類中都定義類似上述結構的方法。當項目中的枚舉類較多時,顯得代碼冗余。

 

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();
}
View Code

  枚舉類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;
  }

}
View 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 }
View Code

  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 }
View Code

  定義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 }
View Code

  最后編寫測試語句:

  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、調用類的靜態變量的時候;


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM