Proxy invocationHandler

动态代理

即,动态代理是利用java反射技术,在运行时创建一个实现某些给定接口的新类。

栗子

先定义接口

public interface people {
    public String work();
}

复制代码

实现该接口

public class Teacher implements people{

    @Override
    public String work() {
        System.out.println("教书育人");
        return "教书";
    }
}

复制代码

编写代理类

import com.sun.xml.internal.ws.api.pipe.FiberContextSwitchInterceptor;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class WorkHandler implements InvocationHandler {
    // 代理类的对象
    private Object obj;

    public WorkHandler(){

    }

    // 在调用这个方法的时候,会被调度到invoke方法,并将参数传入
    public WorkHandler(Object obj){
        this.obj = obj;
    }
    // 在调用给类的方法的时候,会被调度到invoke方法
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        // 真实对象执行之前添加操作
        // 调用obj类的method方法,并且参数为args
        Object invoke = method.invoke(obj, args);
        // 真实对象执行后添加操作
        return invoke;// 返回执行的结果
    }
}

复制代码

最后运行

import java.io.FileInputStream;
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

public class Main {
    public static void main(String[] args){
        // 要代理的对象
        people people = new Teacher();  // 创建要代理的真实对象
        // 进行注册
        InvocationHandler handler = new WorkHandler(people);
        // 进行创建代理对象
        // 定义一个代理对象
        people proxy;
        // 调用Proxy.newProxyInstance方法
        // 需要传入真实对象实现的接口,为下一步调用该方法做准备
        // 将handler关联到invocationHandler对象,用于调度到invoke方法
        // 由于返回的对象为object类型,需要进行强制类型转换,保留people接口定义的方法
        // 最后一个参数要对对象进行关联,最后批量生产出对象
        proxy = (people) Proxy.newProxyInstance(handler.getClass().getClassLoader(), people.getClass().getInterfaces(), handler);
        System.out.println(proxy.work());
    }
}

复制代码

第一个参数

第一个参数是运行时,创建的代理对象。

反射+IO操作读取class文件

import java.io.FileInputStream;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;

public class Main {
    // 需要使用static让其加载进入内存
    static class myClassLoader extends ClassLoader{
        private String classPath;  // 获取当前类的在磁盘中保存的地址
        // 通过构造函数将地址注入
        public myClassLoader(String classPath){
            this.classPath = classPath;
        }
        // 将文件内容加载进入内存
        private byte[] loadByte(String name) throws Exception{
            // 获取一个输入流,
            FileInputStream fis = new FileInputStream(classPath + "/" + name + ".class");
            // 获取长度
            int len = fis.available();
            // 定义byte数组
            byte[] data = new byte[len];
            // 加载进入内存
            fis.read(data);
            // 关闭流
            fis.close();
            return data;
        }
        // 重写findClass方法,让加载的时候调用findClass方法
        protected Class<?> findClass(String name) throws ClassNotFoundException{
            try{
                // 读取文件到数组中
                byte[] data = loadByte(name);
                // 将字节码加载进入内存当中
                return defineClass(name, data, 0, data.length);
            }catch(Exception e){
                e.printStackTrace();
            }
            return null;
        }
    }
    public static void main(String[] args) throws IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, ClassNotFoundException {
        // 先初始化该类
        myClassLoader classLoader = new myClassLoader("/home/ming");
        // 此时会调用findClass加载Test.class加载进入内存当中
        Class clazz = classLoader.loadClass("com.ming.Test");
        // 实例化该类对象
        Object obj = clazz.newInstance();
        // 获取clazz该类方法中名称为hello,参数为空的方法。
        Method helloMethod = clazz.getDeclaredMethod("helloWorld", null);
       // 调用该方法
        // 调用obj类中的helloMethod,参数为空的方法。
       helloMethod.invoke(obj, null);
    }
}

复制代码

转载于:https://juejin.im/post/5c4631d4518825251077db66

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值