Java---注解、类加载器-加强-实现运行任意目录下class中加了@MyTest的空参方法

简介: 做自己的类加载器 虚拟机的核心是通过类加载器来加载.class文件,然后进行相应的解析执行。那么我们可以自己做类加载器,手动加载需要的.class以进行解析执行,从而扩展虚拟机的功能。

做自己的类加载器

虚拟机的核心是通过类加载器来加载.class文件,然后进行相应的解析执行。那么我们可以自己做类加载器,手动加载需要的.class以进行解析执行,从而扩展虚拟机的功能。

以下内容摘自API文档:

应用程序需要实现 ClassLoader 的子类,以扩展 Java 虚拟机动态加载类的方式。

网络类加载器子类必须定义方法 findClass 和 loadClassData,以实现从网络加载类。下载组成该类的字节后,它应该使用方法 defineClass 来创建类实例。

代码示例:
自己的类加载器 MyClassLoader

package cn.hncu;

import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;

import org.junit.Test;

public class MyClassLoader extends ClassLoader{
    public Class<?> findClass(String name){
        //name = "e:\\cn\\hncu\\Person.class"
        Class c = null;
        FileInputStream in;
        byte[] b=null;

        //通过IO或网络把字节码数据读取到buf[]当中。进一步地,
        //如果我们自己熟悉字节码的生成格式,那么也可自己用程序生成。
        //本例,我们是把硬盘中的一个外部字节码文件的数据读取到buf[]当中
        //1
        try {
            in = new FileInputStream(name);
            byte[] buf = new byte[1024]; 
            ByteArrayOutputStream baos = new ByteArrayOutputStream();//字节流
            int len=0;
            while((len=in.read(buf))!=-1){
                baos.write(buf, 0, len);
            }
            in.close();
            baos.close();
            b = baos.toByteArray();
        //2 ---1-2这里可以抽取出来写一个loadClassData方法
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        c = defineClass("cn.hncu.Person", b, 0, b.length);
        return c;
    }

    @Test
    public void testClassData() throws ReflectiveOperationException{
        String className="cn.hncu.Person";
        //用Java的类加载器加载一个
        Class c = Class.forName(className);
        Object obj = c.newInstance();
        System.out.println(obj);
        System.out.println((Person)obj);

        System.out.println("-------------------");
        className = "e:\\cn\\hncu\\Person.class";
        Class c2 = findClass(className);
        Object obj2 = c2.newInstance();
        System.out.println(obj2);

        System.out.println((Person)obj2);//这句是有问题的
        //※不同类加载器加载的对象是无法强转---可以理解是不同的生存空间
        //Person p2 = (Person) obj2;//会挂的。
        //因为obj2的生存空间是MyClassLoader,而Person的生成空间是AppClassLoader
        //System.out.println(p2);


    }


}

测试结果:

看,最后那句不能输出吧。
因为不是一个类加载器的。

作自己的测试工具MyJUnit
(注解与反射共同使用的案例 )

相关说明:

1)JUnit用的是@Test注解,我们用@MyTest注解。

2)JUnit已经嵌入到MyEclipse当中,我们自己的MyJUnit只要能独立运行就可以(不嵌入),同时这样我们也不方便在MyJUnit中以参数方式接收到被测试类的类名与方法名,只能以键盘输入的方式接收。

3)JUnit能实现指定单个方法来调用执行,由于不能利用MyEclipse传参,因此我们在MyJUnit程序中遍历所有的方法并通过判断是否声明@MyTest注解来决定是否调用执行该方法。

下面实现了运行任意目录下的实现了@MyTest注解的方法:
需要输入绝对路径名和类的完整名字。

注解:@MyTest

package cn.hncu.myJunit;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Retention(RetentionPolicy.RUNTIME)//运行时也存在,必须要加这个
@Target (ElementType.METHOD)//限制注解只能加在方法上
public @interface MyTest {

}

测试类:TestPerson

package cn.hncu.myJunit;
/**
 * 测试用的
 * @author 陈浩翔
 *
 * @version 1.0  2016-5-6
 */
public class TestPerson {

    public void run1(){
        System.out.println("run1...");
    }

    @MyTest
    public void run2(){
        System.out.println("run2...");
    }

    public void run3(){
        System.out.println("run3...");
    }

    @MyTest
    public void run4(){
        System.out.println("run4...");
    }

    public void run5(){
        System.out.println("run5...");
    }

}

MyClassLoader类:自己写的类加载器

package cn.hncu.myJunit;

import java.io.ByteArrayOutputStream;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.IOException;

/**
 * 自己写的类加载器
 * @author 陈浩翔
 *
 * @version 1.0  2016-5-6
 */
public class MyClassLoader extends ClassLoader{

//我把它分成2个方法写了。
    public  Class<?> findClass(String name, String className) {
        try {
            byte b[] = loadClassData(name);
            Class c = defineClass(className, b, 0, b.length);
            return c;
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        }
        return null;
    }

    private static byte[] loadClassData(String name) throws IOException {
        byte buf[] = new byte[1024];
        FileInputStream in = new FileInputStream(name);
        ByteArrayOutputStream out = new ByteArrayOutputStream();
        int len=0;
        while((len=in.read(buf))!=-1){
            out.write(buf, 0, len);
        }
        in.close();
        out.close();
        byte b[] = out.toByteArray();
        return b;
    }
}

main方法类:

package cn.hncu.myJunit;

import java.lang.reflect.Method;
import java.util.Scanner;

import cn.hncu.myJunit.MyClassLoader;

/**
 * @author 陈浩翔
 * @version 1.0  2016-5-6
 */
public class MyJunit {

    public static void main(String[] args) throws ReflectiveOperationException {
        Scanner sc = new Scanner(System.in);
        System.out.println("请输入需要运行的类的绝对路径(路径中不能有空格,需要类的.class文件):");
        String name = sc.next();
        System.out.println("请输入类的名称(包含包名):");
        String className = sc.next();
        Class c = (new MyClassLoader()).findClass(name, className);
        //获得那个类了。

        //那个类必须要有空参构造方法
        Object obj = c.newInstance();

        //获得这个类所有声明的方法,包括私有的
        Method ms[] = c.getDeclaredMethods();
        for(Method m:ms){
            if(m.isAnnotationPresent(MyTest.class)){
                m.invoke(obj, null);
            }
        }
    }
}

运行测试结果:

现在我把class文件移动到D盘了。

再看运行结果:

这个可以有很多改进的地方,就比如每次输入路径都很麻烦,
我们可以做一个图形界面,让我们自己选择。
这样就方便多了。

目录
相关文章
|
5天前
|
Java 关系型数据库 MySQL
Elasticsearch【问题记录 01】启动服务&停止服务的2类方法【及 java.nio.file.AccessDeniedException: xx/pid 问题解决】(含shell脚本文件)
【4月更文挑战第12天】Elasticsearch【问题记录 01】启动服务&停止服务的2类方法【及 java.nio.file.AccessDeniedException: xx/pid 问题解决】(含shell脚本文件)
33 3
|
2天前
|
Java
Java 与垃圾回收有关的方法
Java 与垃圾回收有关的方法
|
2天前
|
Java
网页运行java程序cheerpj
网页运行java程序cheerpj
27 0
|
3天前
|
存储 Java 测试技术
一文搞清楚Java中的方法、常量、变量、参数
在JVM的运转中,承载的是数据,而数据的一种变现形式就是“量”,量分为:**常量与变量**,我们在数学和物理学中已经接触过变量的概念了,在Java中的变量就是在程序运行过程中可以改变其值的量。
14 0
|
3天前
|
Java 测试技术 编译器
JAVA注解
JAVA注解
9 0
|
7天前
|
存储 Java
Java动态转发代理IP的实现方法
Java动态转发代理IP的实现方法
23 11
|
8天前
|
Java
Java接口中可以定义哪些方法?
【4月更文挑战第13天】
14 0
Java接口中可以定义哪些方法?
|
14天前
|
Java Shell
Java 21颠覆传统:未命名类与实例Main方法的编码变革
Java 21颠覆传统:未命名类与实例Main方法的编码变革
13 0
|
5月前
|
Java
JAVA 抽象类(Abstract Class) 和 接口(Interface) 的区别
对于面向对象编程来说,抽象是它的一大特征之一。在 Java 中,可以通过两种形式来体现 OOP 的抽象:接口和抽象类。这两者有太多相似的地方,又有太多不同的地方。今天我们就一起来学习一下Java中的接口和抽象类抽象类不能用于实例化对象,抽象类往往用来表示抽象概念。举个例子,中国人(Chinese 类)和美国人(American 类)都有“吃饭”这个行为,因此可以先定义一个 Person 类,然后让 Chinese 和 American 都继承这个类。但如何在父类 Person 中定义“吃饭”这个方法呢?一般
74 0
|
安全 Java API
java基础 之 从Class.forName()跟.class的区别看类的初始化
java基础 之 从Class.forName()跟.class的区别看类的初始化
81 0