java动态代理

动态代理

参考Java动态代理机制详解(JDK 和CGLIB,Javassist,ASM)

代理的实质是在运行期间手动创建class类,对被代理对象的方法进行代理,调用被代理对象的方法,动态代理就是动态的创建Proxy对象,用完之后销毁class类,避免冗杂,动态代理的实现方式主要有以下两种。
CGLIB代理主要是对指定的类生成一个子类,覆盖其中的所有方法,所以该类或方法不能声明称final的。

一、通过jdk实现InvocationHandler实现动态代理
1、定义接口
package cn.zlz.proxy.jdk;

public interface IComputorService {

    /**
     * 卖电脑
     * @param brand
     */
    public void sellComputor(String brand);
    /**
     * 修电脑
     */
    public void repairComputor(String brand);

}
2、定义接口实现类
package cn.zlz.proxy.jdk;

public class ThinkPadSeller implements IComputorService{

    public void sellComputor(String brand) {
        System.out.println("sell the thinkPad computor");

    }

    public void repairComputor(String brand) {
        System.out.println("repair the thinkPad computor");
    }

}
3、定义生成代理对象
package cn.zlz.proxy.jdk;

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

public class SimpleProxyImpl implements InvocationHandler {

    // 被代理对象
    private ThinkPadSeller thinkPadSeller;

    public SimpleProxyImpl(ThinkPadSeller thinkPadSeller) {
        super();
        this.thinkPadSeller = thinkPadSeller;
    }

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("代理开始");
        // 调用被代理对象
        method.invoke(thinkPadSeller, args);
        System.out.println("代理结束");
        return null;
    }
    //提供方法获取代理对象  
    public IComputorService newProxy(){  

        //使用Proxy类创建代理对象  
        IComputorService proxyInstance = (IComputorService) Proxy.newProxyInstance(thinkPadSeller.getClass().getClassLoader(), //使用被代理对象的加载器  
                thinkPadSeller.getClass().getInterfaces(), //使用被代理对象的接口  
                this );//匿名内部类比较坑,所以我们找一个类实现并覆写方法,直接用本类,现成的..  
        return proxyInstance;  
    }  
}
4、main函数测试
package cn.zlz.proxy.jdk;

import java.lang.reflect.Proxy;

/**
 *  通过jdk的实现invocationHandler接口只能代理实现接口的对象
 *  为了解决这个问题,就有了动态地创建Proxy的想法:在运行状态中,需要代理的地方,根据接口 和被代理对象,
 *  动态地创建一个Proxy,用完之后,就会销毁,这样就可以避免了Proxy 角色的class在系统中冗杂的问题了。
 *
 */
public class Main {

    public static void main(String[] args) {
        /**
         * 使用Proxy创建代理对象
         * 1、被代理对象
         * 2、被代理对象实现的接口s
         * 3、Invocation实现对象
         */
        //使用Proxy类创建代理对象
        Class beProxyClazz = ThinkPadSeller.class;
        ThinkPadSeller thinkPadSeller = new ThinkPadSeller();
        ClassLoader classLoader = beProxyClazz.getClassLoader();
        Class[] interfaces = beProxyClazz.getInterfaces();
        SimpleProxyImpl simpleProxyImpl = new SimpleProxyImpl(thinkPadSeller);
        // 根据上面提供的信息,创建代理对象 在这个过程中,JDK会通过根据传入的参数信息动态地在内存中创建和.class 文件等同的字节码 ,然后根据相应的字节码转换成对应的class,然后调用newInstance()创建实例 
        IComputorService proxy = (IComputorService) Proxy.newProxyInstance(classLoader, interfaces, simpleProxyImpl);
        /*
         * 生成的代理对象编译后的代码为 public final repairComputor(){this.h.invoke(this, m3, null);m3 = Class.forName("cn.zlz.proxy.jdk.ThinkPadSeller").getMethod("repairComputor", [String.class]); }
         * this指的是invocation的实现类,调用invoke方法,并将被代理对象的方法作为参数传递
         */
        proxy.repairComputor("thinkPad");
    }
}
而、通过cglib实现动态代理
1、定义被代理对象
package cn.zlz.proxy.cglib;

public class ThinkPadSeller {

    public void sellComputor(String brand) {
        System.out.println("sell the thinkPad computor");

    }

    public void repairComputor(String brand) {
        System.out.println("repair the thinkPad computor");
    }

}
2、实现cglib的MethodInterceptor
package cn.zlz.proxy.cglib;

import java.lang.reflect.Method;

import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;

/*
 * cglib代理,实现MethodInterceptor
 */
public class CglibProxy implements MethodInterceptor {

    public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
        System.out.println("开始代理");
        //代理类是继承的被代理类,调用父类的原方法
        proxy.invokeSuper(obj, args);  

        System.out.println("结束代理");
        return null;
    }

}
3、main函数调试
package cn.zlz.proxy.cglib;

import net.sf.cglib.proxy.Enhancer;

/*
 * 代理对象继承被代理对象
  1.查找被代理类的所有非final 的public类型的方法定义;
  2.将这些方法的定义转换成字节码;
  3.将组成的字节码转换成相应的代理的class对象;
  4.实现 MethodInterceptor接口,用来处理 对代理类上所有方法的请求(这个接口和JDK动态代理InvocationHandler的功能和角色是一样的)
 */
public class Main {
    public static void main(String[] args) {
        CglibProxy cglibProxy = new CglibProxy();
        // cglib 中加强器,用来创建动态代理
        Enhancer enhancer = new Enhancer();
        // 设置要创建动态代理的类,即父类
        enhancer.setSuperclass(ThinkPadSeller.class);
        // 设置回调,这里相当于是对于代理类上所有方法的调用,都会调用CallBack,而Callback则需要实行intercept()方法进行拦截
        enhancer.setCallback(cglibProxy);
        ThinkPadSeller proxy = (ThinkPadSeller) enhancer.create();
        proxy.repairComputor("thinkpad");
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值