码迷,mamicode.com
首页 > 编程语言 > 详细

java扫描某个包下的所有java类并加载

时间:2019-09-01 12:47:39      阅读:331      评论:0      收藏:0      [点我收藏+]

标签:@param   scan   扫描   row   bst   lis   tabs   eclipse   not   

  最近在学习java的反射和注解,实际情景中需要扫描某个包下的所有java类,然后使用类加载器加载类。

  基本思路,获得程序的路径扫描src下某个包内的子包和java类,实现也比较简单。

  运行环境:windows10+jdk1.8+eclipse

  直接贴代码

  

package org.test.scanner;

import java.io.File;
import java.util.ArrayList;
import java.util.List;

/*
 * date:2019-07-23
 * */
public class PackageScanner {
    private List<Class<?>> classes;
    private String packagePath = null;

    /*
     * 无参构造方法,内部调用带参的构造方法。
     * 
     * @throw classNotFound
     * 
     */
    public PackageScanner() throws ClassNotFoundException {
        this("");
    }

    /*
     * 实现,调用fileScanner进行目录扫描和加载
     * 
     * @param String 传入需要扫描的包
     * 
     * @throw classNotFound
     */
    public PackageScanner(String basePackage) throws ClassNotFoundException {
        packagePath = System.getProperty("user.dir") + "\\src\\";
        String filePath = packagePath + basePackage.replace(‘.‘, ‘\\‘);
        classes = new ArrayList<Class<?>>();
        fileScanner(new File(filePath));
    }

    private void fileScanner(File file) throws ClassNotFoundException {
        if (file.isFile() && file.getName().lastIndexOf(".java") == file.getName().length() - 5) {//5是".java"的长度
            String filePath = file.getAbsolutePath();
            String qualifiedName = filePath.substring(packagePath.length(), filePath.length() - 5).replace(‘\\‘, ‘.‘);
            System.out.println(qualifiedName);
            classes.add(Class.forName(qualifiedName));
            return;
        } else if (file.isDirectory()) {
            for (File f : file.listFiles())
                fileScanner(f);
        }
    }

    /*
     * 得到加载到的类对象的List,返回的是ArrayList
     */
    public List<Class<?>> getClasses() {
        return this.classes;
    }
}

  

这是一个简单的包扫描类,这里直接使用Class.forName()加载扫描到的类

我们可以看一下forName实现

 

public static Class<?> forName(String className) throws ClassNotFoundException {
	return forName0(className, true, ClassLoader.getCallerClassLoader());
	}

 

发现调用了ClassLoader.getCallerClassLoader() 

从名字上可以看出是得到调用类的类加载器,我们可以看一下它的实现

 

static ClassLoader getCallerClassLoader() {
		// NOTE use of more generic Reflection.getCallerClass()
		Class caller = Reflection.getCallerClass(3);
		// This can be null if the VM is requesting it
		if (caller == null) {
			return null;
		}
		return caller.getClassLoader0();
	}

 

关键一句: Reflection.getCallerClass(3)。

一直往上传递,直到获取到它的调用类,然后得到调用类的类加载器


其中 REflection.getCallerClass()的参数有:

0 和小于0  -   返回 Reflection类 

1  -   返回自己的类 

2  -    返回调用者的类 
3. 4. ....层层上传。

 

最后的目的就是谁调用这个类,调用类的类加载器就负责加载这个类。只有当它的加载类为null时,即没有任何加载器可用时,才使用getClassLoader0()这个native方法,这是启动类加载器的实现方法,如果不是java lib目录里的库,该类是不会被加载的。

 

通过学习java 的包扫描和类加载,我简单的了解了java类加载器的用法。

能力有限,如有错误请告知一声。

 

 

ps:学而不思则罔,思而不学则殆。

java扫描某个包下的所有java类并加载

标签:@param   scan   扫描   row   bst   lis   tabs   eclipse   not   

原文地址:https://www.cnblogs.com/lingdurebing/p/ldrb-java.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!