动态代理类的源码是程序在运行期间由JVM根据反射等机制动态生成的,所以不存在代理类的字节码文件。代理角色和真实角色的联系在程序运行时确定!

Java中有两种动态代理,一种是JDK自带的,另一种的CGLIB动态代理。

jdk方式,委托类必须实现接口,代理类只能对接口进行代理。使用java的反射机制,以及Proxy和InvocationHandler来实现,代理类与委托类实现了相同的接口。

CGLIB(code generate library),代理类可对类进行代理,使用第三方cglib库来实现,其内部使用asm框架生成代理类的字节码,其字节码文件更加复杂,不能代理final方法,因为代理类是委托类的子类。

下面介绍的是JDK自带的动态代理:

代码结构:

 

/**
 * 抽象角色,真实对象和代理对象共同的接口
 * @author Administrator
 *
 */
public interface IUserInfo {
    public void queryUser();

}

 

/**
 * 真实角色
 * @author Administrator
 *
 */
public class JdkDynamicProxy implements IUserInfo{

    @Override
    public void queryUser() {
        System.out.println("真实角色查询方法");
    }

    
}
/**
 * 代理角色处理器
 * 
 * @author Administrator
 *
 */
public class UserHandler implements InvocationHandler {
    private Object target;//真实对象

    public Object bind(Object target) {//代理对象和真实对象建立关系
        this.target = target;
                                    //调用真实对象类加载器                获取真实对象实现了哪些接口,让代理对象下挂在这些接口中   当前对象
        return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);

    }

    @Override // 代理对象 真实对象方法 调用方法的参数
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("真实方法前做一些事情");
        Object object = method.invoke(target, args);// 这里调用的就是真实方法,target对象是通过反射调用
        System.out.println("真实方法后做一些事情");
        return object;
    }

}
/**
 * 调用
 * @author Administrator
 *
 */
public class Test {
    public static void main(String[] args) {
        IUserInfo userImpl = new JdkDynamicProxy();//创建真实对象
        UserHandler handler = new UserHandler();//绑定,生成代理对象
        IUserInfo userInfo = (IUserInfo) handler.bind(userImpl);
        userInfo.queryUser();
    }
}

 

posted on 2018-03-26 14:55  帅过驴的袋鼠  阅读(122)  评论(0编辑  收藏  举报