java动态代理实现接口调用

动态代理是基于接口实现的代理,mybatis就是用这个技术实现的

首先,创建一个最基本接口和实现类:

interface UserManager {
    void addUser();
}

class UserManagerImpl implements UserManager{

    @Override
    public void addUser() {
        System.out.println("添加用户!!");
    }
}

关键:代理类

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

/**
 * 动态代理类只能代理接口(不支持抽象类),代理类都需要实现InvocationHandler类,
 * 实现invoke方法。该invoke方法就是调用被代理接口的所有方法时需要调用的,
 * 该invoke方法返回的值是被代理接口的一个实现类
 */
 class DynamicProxyUtil implements InvocationHandler {

    /**
     * 代理对象,这里是实现类,然后通过下面的Instance返回接口的引用
     */
    private Object obj;

    public DynamicProxyUtil(Object obj) {
        this.obj = obj;
    }

    /**
     * 绑定关系,也就是关联到哪个接口(与具体的实现类绑定)的哪些方法将被调用时,执行invoke方法。
     * @return      动态代理的接口对象
     */
    public Object Instance(){
        //该方法用于为指定类装载器、一组接口及调用处理器生成动态代理类实例
        //第一个参数指定产生代理对象的类加载器,需要将其指定为和目标对象同一个类加载器
        //第二个参数要实现和目标对象一样的接口,所以只需要拿到目标对象的实现接口
        //第三个参数表明这些被拦截的方法在被拦截时需要执行哪个InvocationHandler的invoke方法
        //根据传入的目标返回一个代理对象
        return Proxy.newProxyInstance(obj.getClass().getClassLoader(),
                obj.getClass().getInterfaces(),this);
    }

    /**
     *  关联的这个实现类的方法被调用时将被执行
     * @param proxy     表示代理
     * @param method    表示原对象被调用的方法
     * @param args      表示方法的参数
     * @return          这里就通过动态代理调用方法了
     * @throws Throwable
     */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args)
            throws Throwable {
        return method.invoke(obj, args);
    }

}

测试:

 public static void main(String[] args){
        DynamicProxyUtil proxy = new DynamicProxyUtil(new UserManagerImpl());
        //获得代理对象
        UserManager p =(UserManager) proxy.Instance();
        //调用代理对象的sing方法
        p.addUser();
    }

动态代理成功。 

猜你喜欢

转载自blog.csdn.net/qq_39648029/article/details/102544360