클래스의 모든 서브 클래스 경로 지정된 모든 패키지를 가져옵니다

1. 문제

개발 과정, 그리고 때로는 모든 클래스 패스, 클래스의 모든 서브 클래스에서 특정 패키지를 찾을 필요가, 어떻게 할까?

구현 (2)

더 일반적인 솔루션은 디렉토리 자체를 통과하는 모든의 .class 파일을 찾을 수 있습니다.
이 방법은 프로세스를 단순화하기 위해 구현 다음 봄 도구는 더 이상 자신의 디렉토리 탐색을 필요로 사용하지 않습니다

    /**
     * 获取在指定包下某个class的所有非抽象子类
     *
     * @param parentClass 父类
     * @param packagePath 指定包,格式如"com/sinosun/tarvel"
     * @return 该父类对应的所有子类列表
     */
    private static <E> List<Class<E>> getSubClasses(final Class<E> parentClass, final String packagePath) throws ClassNotFoundException
    {
        final ClassPathScanningCandidateComponentProvider provider = new ClassPathScanningCandidateComponentProvider(false);
        provider.addIncludeFilter(new AssignableTypeFilter(parentClass));
        final Set<BeanDefinition> components = provider.findCandidateComponents(packagePath);
        final List<Class<E>> subClasses = new ArrayList<>();
        for (final BeanDefinition component : components)
        {
            @SuppressWarnings("unchecked") final Class<E> cls = (Class<E>) Class.forName(component.getBeanClassName());
            if (Modifier.isAbstract(cls.getModifiers()))
            {
                continue;
            }
            subClasses.add(cls);
        }
        return subClasses;
    }

3. 예

스캔 com.sinosun모든 패키지 CoreStart서브 클래스 및 서브 클래스에 의해 순차적으로 반사하는 데이터 소스를 활성화한다.

package com.sinosun.travel.core.main;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider;
import org.springframework.core.type.filter.AssignableTypeFilter;

import java.lang.reflect.Constructor;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
import java.util.ArrayList;
import java.util.List;
import java.util.Set;

/**
 * @author caogu
 * @date 2019/5/31 13:36
 */
public class DataSourceHandler
{
    private static final Logger logger = LoggerFactory.getLogger(DataSourceHandler.class);
    private static final String PARENT_PACKAGE= "com.sinosun";
    private static final String CORE_START_CLASS_NAME = "com.sinosun.travel.core.main.CoreStart";
    private static final String IS_ENABLE_DATA_SOURCE_METHOD_NAME = "isEnableDataSource";

    /**
     * 根据环境判定是否启用数据源
     * 默认不启用数据源;扫描CoreStart类的所有子类,若子类中有一个启用数据源则启用数据源
     * @throws Exception 异常
     */
    public static void scanIsEnableDataSource() throws Exception
    {
        List<Class<CoreStart>> coreStartClasses = getSubClasses(CoreStart.class, PARENT_PACKAGE);
        logger.info("扫描到CoreStart类的子类有:{}", coreStartClasses);

        for (Class<CoreStart> coreStartClass : coreStartClasses)
        {
            // 只扫描CoreStart的子类,只要一个启用数据源就启用数据源; 默认不启用数据源
            // 只检测子类,父类排除
            if (!CORE_START_CLASS_NAME.equals(coreStartClass.getName()))
            {
                Constructor constructor = coreStartClass.getConstructor();
                Object obj = constructor.newInstance();

                Method[] methods = coreStartClass.getDeclaredMethods();
                for (Method method : methods)
                {
                    if (IS_ENABLE_DATA_SOURCE_METHOD_NAME.equals(method.getName()))
                    {
                        boolean isEnableDataSource = (boolean) method.invoke(obj);
                        logger.info("调用{}.{}方法的返回值为{}", coreStartClass.getName(), coreStartClass.getName(), isEnableDataSource);
                        if (isEnableDataSource)
                        {
                            CoreStart.ENABLE_DATA_SOURCE = true;
                            logger.info("扫描到子类启用数据源,加载数据源!");
                            return;
                        }
                    }
                }
            }
        }
    }

    /**
     * 获取在指定包下某个class的所有非抽象子类
     *
     * @param parentClass 父类
     * @param packagePath 指定包,格式如"com/sinosun/tarvel"
     * @return 该父类对应的所有子类列表
     */
    private static <E> List<Class<E>> getSubClasses(final Class<E> parentClass, final String packagePath) throws ClassNotFoundException
    {
        final ClassPathScanningCandidateComponentProvider provider = new ClassPathScanningCandidateComponentProvider(false);
        provider.addIncludeFilter(new AssignableTypeFilter(parentClass));
        final Set<BeanDefinition> components = provider.findCandidateComponents(packagePath);
        final List<Class<E>> subClasses = new ArrayList<>();
        for (final BeanDefinition component : components)
        {
            @SuppressWarnings("unchecked") final Class<E> cls = (Class<E>) Class.forName(component.getBeanClassName());
            if (Modifier.isAbstract(cls.getModifiers()))
            {
                continue;
            }
            subClasses.add(cls);
        }
        return subClasses;
    }

    public static void main(String[] args)
    {
        System.out.println(System.getProperty("java.class.path"));
        System.out.println(System.getProperty("user.dir"));
    }

}

결과는 다음과 같다 :
그림 삽입 설명 여기

게시 된 418 개 원래 기사 · 원의 찬양 (745) · 조회수 126 만 +

추천

출처blog.csdn.net/u013467442/article/details/90758487