JAVA动态代理


因为现在在在学java的一些框架技术,比如ssh,mybatis等,总是提到动态代理,而且用到的还蛮多的,因此将jdk动态代理好好研究一下。

代理模式
代理模式是java中一种常见的设计模式,他的特征是代理类和委托类实现相同的接口。代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。java动态代理的出现,使我们不需要手动写代理类,只需要提供一组接口和委托类对象,即可动态生成代理类。

CGLIB动态代理

CGLIB代理是针对类实现代理,主要是指定的类生成一个子类,覆盖其中的所有方法,所有该类或方法不能声明称final的。

JDK动态代理

代理对象和目标对象实现了相同的接口,目标对象作为代理代理对象的一个属性,可以在调用目标对象响应方法前后加上其他业务的逻辑。JDK动态代理只能针对实现了接口的类生成代理。

CGLIB测试代码:

定义一个实现MethodInterceptor的方法:


package com.jcw.proxy;

import net.sf.cglib.proxy.Enhancer;
import net.sf.cglib.proxy.MethodInterceptor;
import net.sf.cglib.proxy.MethodProxy;
import java.lang.reflect.Method;
public class DynamicProxyByCGLib implements MethodInterceptor {

private Object target;

public DynamicProxyByCGLib(Object target) {
    this.target = target;
}

public Object getInstance() {
    Enhancer enhancer = new Enhancer();
    enhancer.setSuperclass(this.target.getClass());
    enhancer.setCallback(this);
    return enhancer.create();
}

public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
    System.out.println("before intercept");
    methodProxy.invokeSuper(o, objects);
    System.out.println("after interept");
    return null;
}

}

定义一个测试类:


package com.jcw.proxy;
public class CGLIBProxyTest {
    public void say() {
        System.out.println("say for CGLIBProxyTest");
    }
    public static void main(String[] args) {
        CGLIBProxyTest cglibProxyTest = new CGLIBProxyTest();
        DynamicProxyByCGLib dynamicProxyByCGLib = new DynamicProxyByCGLib(cglibProxyTest);
        CGLIBProxyTest cglibProxyTest1 = (CGLIBProxyTest) dynamicProxyByCGLib.getInstance();
        cglibProxyTest1.say();
    }
}

JDK测试代码:

定义一个接口:

package com.jcw.proxy;
public interface JdkInterfaces1 {
    void say();
}
定义一个实现InvocationHandler接口的类:

package com.jcw.proxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.util.Arrays;
public class DynamicProxyByJDK implements InvocationHandler {
    private Object target;

public DynamicProxyByJDK(Object target) {
    this.target = target;
}

public Object getInstance() {
    System.out.println("classLoader:" + target.getClass().getClassLoader());
    System.out.println("interfaces:" + Arrays.toString(target.getClass().getInterfaces()));
    return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
}
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    System.out.println("before invoke");
    method.invoke(target, args);
    System.out.println("after invoke");
    return null;
}

}

jdk测试的类


package com.jcw.proxy;
public class JdkProxyTest implements JdkInterfaces1{

public void say() {
    System.out.println("say for JdkProxyTest");
}

public static void main(String[]  args) {
    JdkProxyTest jdkProxyTest = new JdkProxyTest();
    DynamicProxyByJDK dynamicProxyByJDK = new DynamicProxyByJDK(jdkProxyTest);
    JdkInterfaces1 jdkProxyTest1 = (JdkInterfaces1) dynamicProxyByJDK.getInstance();
    jdkProxyTest1.say();
}

}

总结:CGLIB主要是针对类实现代理,主要通过根据指定的类生成一个代理类,覆盖其中的方法。当然对于实现接口的类,CGLIB同样可以实现。
JDK动态代理只能对实现了接口的类生成代理,而不是针对类。

  • 2
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值