package com.alibaba.excel.util; import java.lang.reflect.Field; import java.lang.reflect.Modifier; import java.util.ArrayList; import java.util.Collections; import java.util.HashMap; import java.util.HashSet; import java.util.LinkedHashSet; import java.util.List; import java.util.Map; import java.util.Optional; import java.util.TreeMap; import java.util.concurrent.ConcurrentHashMap; import com.alibaba.excel.annotation.ExcelIgnore; import com.alibaba.excel.annotation.ExcelIgnoreUnannotated; import com.alibaba.excel.annotation.ExcelProperty; import com.alibaba.excel.annotation.write.style.ContentFontStyle; import com.alibaba.excel.annotation.write.style.ContentStyle; import com.alibaba.excel.converters.AutoConverter; import com.alibaba.excel.converters.Converter; import com.alibaba.excel.exception.ExcelCommonException; import com.alibaba.excel.metadata.Holder; import com.alibaba.excel.metadata.property.ExcelContentProperty; import com.alibaba.excel.metadata.property.FontProperty; import com.alibaba.excel.metadata.property.StyleProperty; import com.alibaba.excel.write.metadata.holder.WriteHolder; import net.sf.cglib.beans.BeanMap; /** * Class utils * * @author Jiaju Zhuang **/ public class ClassUtils { public static final Map, FieldCache> FIELD_CACHE = new ConcurrentHashMap<>(); /** * The cache configuration information for each of the class */ public static final Map, Map> CLASS_CONTENT_CACHE = new ConcurrentHashMap<>(); /** * The cache configuration information for each of the class */ public static final Map CONTENT_CACHE = new ConcurrentHashMap<>(); /** * Calculate the configuration information for the class * * @param dataMap * @param headClazz * @param fieldName * @return */ public static ExcelContentProperty declaredExcelContentProperty(Map dataMap, Class headClazz, String fieldName) { Class clazz = null; if (dataMap instanceof BeanMap) { Object bean = ((BeanMap)dataMap).getBean(); if (bean != null) { clazz = bean.getClass(); } } return getExcelContentProperty(clazz, headClazz, fieldName); } private static ExcelContentProperty getExcelContentProperty(Class clazz, Class headClass, String fieldName) { return CONTENT_CACHE.computeIfAbsent(buildKey(clazz, headClass, fieldName), key -> { ExcelContentProperty excelContentProperty = Optional.ofNullable(declaredFieldContentMap(clazz)) .map(map -> map.get(fieldName)) .orElse(null); ExcelContentProperty headExcelContentProperty = Optional.ofNullable(declaredFieldContentMap(headClass)) .map(map -> map.get(fieldName)) .orElse(null); ExcelContentProperty combineExcelContentProperty = new ExcelContentProperty(); combineExcelContentProperty(combineExcelContentProperty, headExcelContentProperty); if (clazz != headClass) { combineExcelContentProperty(combineExcelContentProperty, excelContentProperty); } return combineExcelContentProperty; }); } public static void combineExcelContentProperty(ExcelContentProperty combineExcelContentProperty, ExcelContentProperty excelContentProperty) { if (excelContentProperty == null) { return; } if (excelContentProperty.getField() != null) { combineExcelContentProperty.setField(excelContentProperty.getField()); } if (excelContentProperty.getConverter() != null) { combineExcelContentProperty.setConverter(excelContentProperty.getConverter()); } if (excelContentProperty.getDateTimeFormatProperty() != null) { combineExcelContentProperty.setDateTimeFormatProperty(excelContentProperty.getDateTimeFormatProperty()); } if (excelContentProperty.getNumberFormatProperty() != null) { combineExcelContentProperty.setNumberFormatProperty(excelContentProperty.getNumberFormatProperty()); } if (excelContentProperty.getContentStyleProperty() != null) { combineExcelContentProperty.setContentStyleProperty(excelContentProperty.getContentStyleProperty()); } if (excelContentProperty.getContentFontProperty() != null) { combineExcelContentProperty.setContentFontProperty(excelContentProperty.getContentFontProperty()); } } private static String buildKey(Class clazz, Class headClass, String fieldName) { String key = ""; if (clazz != null) { key += clazz.getName(); } key += "-"; if (headClass != null) { key += headClass.getName(); } key += "-"; if (fieldName != null) { key += fieldName; } return key; } private static Map declaredFieldContentMap(Class clazz) { if (clazz == null) { return null; } return CLASS_CONTENT_CACHE.computeIfAbsent(clazz, key -> { List tempFieldList = new ArrayList<>(); Class tempClass = clazz; while (tempClass != null) { Collections.addAll(tempFieldList, tempClass.getDeclaredFields()); // Get the parent class and give it to yourself tempClass = tempClass.getSuperclass(); } ContentStyle parentContentStyle = clazz.getAnnotation(ContentStyle.class); ContentFontStyle parentContentFontStyle = clazz.getAnnotation(ContentFontStyle.class); Map fieldContentMap = MapUtils.newHashMapWithExpectedSize( tempFieldList.size()); for (Field field : tempFieldList) { ExcelContentProperty excelContentProperty = new ExcelContentProperty(); excelContentProperty.setField(field); ExcelProperty excelProperty = field.getAnnotation(ExcelProperty.class); if (excelProperty != null) { Class> convertClazz = excelProperty.converter(); if (convertClazz != AutoConverter.class) { try { Converter converter = convertClazz.newInstance(); excelContentProperty.setConverter(converter); } catch (Exception e) { throw new ExcelCommonException( "Can not instance custom converter:" + convertClazz.getName()); } } } ContentStyle contentStyle = field.getAnnotation(ContentStyle.class); if (contentStyle == null) { contentStyle = parentContentStyle; } excelContentProperty.setContentStyleProperty(StyleProperty.build(contentStyle)); ContentFontStyle contentFontStyle = field.getAnnotation(ContentFontStyle.class); if (contentFontStyle == null) { contentFontStyle = parentContentFontStyle; } excelContentProperty.setContentFontProperty(FontProperty.build(contentFontStyle)); fieldContentMap.put(field.getName(), excelContentProperty); } return fieldContentMap; }); } /** * Parsing filed in the class * * @param clazz Need to parse the class * @param sortedAllFiledMap Complete the map of sorts * @param indexFiledMap Use the index to sort fields * @param ignoreMap You want to ignore field map * @param needIgnore If you want to ignore fields need to ignore * @param holder holder */ public static void declaredFields(Class clazz, Map sortedAllFiledMap, Map indexFiledMap, Map ignoreMap, Boolean needIgnore, Holder holder) { FieldCache fieldCache = declaredFields(clazz); if (fieldCache == null) { return; } if (ignoreMap != null) { ignoreMap.putAll(fieldCache.getIgnoreMap()); } Map tempIndexFieldMap = indexFiledMap; if (tempIndexFieldMap == null) { tempIndexFieldMap = MapUtils.newTreeMap(); } tempIndexFieldMap.putAll(fieldCache.getIndexFiledMap()); Map originSortedAllFiledMap = fieldCache.getSortedAllFiledMap(); if (!needIgnore) { sortedAllFiledMap.putAll(originSortedAllFiledMap); return; } int index = 0; for (Map.Entry entry : originSortedAllFiledMap.entrySet()) { Integer key = entry.getKey(); Field field = entry.getValue(); // The current field needs to be ignored if (((WriteHolder)holder).ignore(entry.getValue().getName(), entry.getKey())) { if (ignoreMap != null) { ignoreMap.put(field.getName(), field); } tempIndexFieldMap.remove(index); } else { // Mandatory sorted fields if (tempIndexFieldMap.containsKey(key)) { sortedAllFiledMap.put(key, field); } else { // Need to reorder automatically // Check whether the current key is already in use while (sortedAllFiledMap.containsKey(index)) { index++; } sortedAllFiledMap.put(index++, field); } } } } public static void declaredFields(Class clazz, Map sortedAllFiledMap, Boolean needIgnore, WriteHolder writeHolder) { declaredFields(clazz, sortedAllFiledMap, null, null, needIgnore, writeHolder); } private static FieldCache declaredFields(Class clazz) { if (clazz == null) { return null; } return FIELD_CACHE.computeIfAbsent(clazz, key -> { List tempFieldList = new ArrayList<>(); Class tempClass = clazz; // When the parent class is null, it indicates that the parent class (Object class) has reached the top // level. while (tempClass != null) { Collections.addAll(tempFieldList, tempClass.getDeclaredFields()); // Get the parent class and give it to yourself tempClass = tempClass.getSuperclass(); } // Screening of field Map> orderFiledMap = new TreeMap>(); Map indexFiledMap = new TreeMap(); Map ignoreMap = new HashMap(16); ExcelIgnoreUnannotated excelIgnoreUnannotated = clazz.getAnnotation(ExcelIgnoreUnannotated.class); for (Field field : tempFieldList) { declaredOneField(field, orderFiledMap, indexFiledMap, ignoreMap, excelIgnoreUnannotated); } return new FieldCache(buildSortedAllFiledMap(orderFiledMap, indexFiledMap), indexFiledMap, ignoreMap); }); } private static Map buildSortedAllFiledMap(Map> orderFiledMap, Map indexFiledMap) { Map sortedAllFiledMap = new HashMap( (orderFiledMap.size() + indexFiledMap.size()) * 4 / 3 + 1); Map tempIndexFiledMap = new HashMap(indexFiledMap); int index = 0; for (List fieldList : orderFiledMap.values()) { for (Field field : fieldList) { while (tempIndexFiledMap.containsKey(index)) { sortedAllFiledMap.put(index, tempIndexFiledMap.get(index)); tempIndexFiledMap.remove(index); index++; } sortedAllFiledMap.put(index, field); index++; } } sortedAllFiledMap.putAll(tempIndexFiledMap); return sortedAllFiledMap; } private static void declaredOneField(Field field, Map> orderFiledMap, Map indexFiledMap, Map ignoreMap, ExcelIgnoreUnannotated excelIgnoreUnannotated) { ExcelIgnore excelIgnore = field.getAnnotation(ExcelIgnore.class); if (excelIgnore != null) { ignoreMap.put(field.getName(), field); return; } ExcelProperty excelProperty = field.getAnnotation(ExcelProperty.class); boolean noExcelProperty = excelProperty == null && excelIgnoreUnannotated != null; if (noExcelProperty) { ignoreMap.put(field.getName(), field); return; } boolean isStaticFinalOrTransient = (Modifier.isStatic(field.getModifiers()) && Modifier.isFinal(field.getModifiers())) || Modifier.isTransient(field.getModifiers()); if (excelProperty == null && isStaticFinalOrTransient) { ignoreMap.put(field.getName(), field); return; } if (excelProperty != null && excelProperty.index() >= 0) { if (indexFiledMap.containsKey(excelProperty.index())) { throw new ExcelCommonException("The index of '" + indexFiledMap.get(excelProperty.index()).getName() + "' and '" + field.getName() + "' must be inconsistent"); } indexFiledMap.put(excelProperty.index(), field); return; } int order = Integer.MAX_VALUE; if (excelProperty != null) { order = excelProperty.order(); } List orderFiledList = orderFiledMap.computeIfAbsent(order, key -> ListUtils.newArrayList()); orderFiledList.add(field); } private static class FieldCache { private final Map sortedAllFiledMap; private final Map indexFiledMap; private final Map ignoreMap; public FieldCache(Map sortedAllFiledMap, Map indexFiledMap, Map ignoreMap) { this.sortedAllFiledMap = sortedAllFiledMap; this.indexFiledMap = indexFiledMap; this.ignoreMap = ignoreMap; } public Map getSortedAllFiledMap() { return sortedAllFiledMap; } public Map getIndexFiledMap() { return indexFiledMap; } public Map getIgnoreMap() { return ignoreMap; } } /** *

Gets a {@code List} of all interfaces implemented by the given * class and its superclasses.

* *

The order is determined by looking through each interface in turn as * declared in the source file and following its hierarchy up. Then each * superclass is considered in the same way. Later duplicates are ignored, * so the order is maintained.

* * @param cls the class to look up, may be {@code null} * @return the {@code List} of interfaces in order, * {@code null} if null input */ public static List> getAllInterfaces(final Class cls) { if (cls == null) { return null; } final LinkedHashSet> interfacesFound = new LinkedHashSet<>(); getAllInterfaces(cls, interfacesFound); return new ArrayList<>(interfacesFound); } /** * Gets the interfaces for the specified class. * * @param cls the class to look up, may be {@code null} * @param interfacesFound the {@code Set} of interfaces for the class */ private static void getAllInterfaces(Class cls, final HashSet> interfacesFound) { while (cls != null) { final Class[] interfaces = cls.getInterfaces(); for (final Class i : interfaces) { if (interfacesFound.add(i)) { getAllInterfaces(i, interfacesFound); } } cls = cls.getSuperclass(); } } }