您的位置:首页 > 职场人生

黑马程序员_(15)类加载器的委托机制及动态代理的三种实现

2015-04-19 07:41 218 查看
------- android培训java培训、期待与您交流! ----------

类加载器的委托机制
1、JVM有多个加载器。类加载器本身也是JAVA类 BootStrap在JVM内核中它并不是JAVA类。系统默认有三个加载器:BootStrap<--ExtClassLoader <--AppClassLoader。
三个加载器的加载管辖范围各有不同
BootStrap-->加载JRE/lib/rt.jar
ExtClassLoader-->加载JRE/lib/ext/*.jar
AppClassLoader-->加载ClassPath指定的jar或目录。
可以定义自己的类加载器。不过定义的类要继承ClassLoader类。
2、加载器的委托机制
       类加载器在尝试自己去查找某个类的字节代码并定义它时,会先代理给其父类加载器,由父类加载器先去尝试加载这个类,依次类推。Java 虚拟机是如何判定两个Java 类是相同的。Java 虚拟机不仅要看类的全名是否相同,还要看加载此类的类加载器是否一样。只有两者都相同的情况,才认为两个类是相同的。即便是同样的字节代码,被不同的类加载器加载之后所得到的类,也是不同的。比如一个 Java 类com.example.Sample,编译之后生成了字节代码文件 Sample.class。两个不同的类加载器 ClassLoaderA 和 ClassLoaderB 分别读取了这个 Sample.class 文件,并定义出两个 java.lang.Class 类的实例来表示这个类。这两个实例是不相同的。对于 Java 虚拟机来说,它们是不同的类。
1、 首先当前线程的类加载器(ContextClassLoader)去加载线程中的第一个类A。
2、 如果类A引用了类B,那么JAVA虚拟机用加载A的类加载器去继续去加载B。
3、 另外还可以调用LoadClass()方法去指定某个类加载器去加载某个类。
类加载器加载类时委托给他的上一级父类加载器,直到最顶端,由它来加载。若在最顶端的那个加载器没有加载到了类再返回到其子类加载器,再由它来执行加载,以此类推,当返回到发起者类加载器时,如果还没找到就抛出异常。

 二、动态代理的三种实现
第一种:
Constructor constructor = clazzProxy.getConstructor(InvocationHandler.class);
  Collection proxy = (Collection) constructor.newInstance(new MyInvocationhandler());
//MyInvocationhandler为实现InvocationHandler接口的类作为参数传入
  System.out.println(proxy);
第二种:
Collection proxy2 = (Collection) constructor .newInstance(new InvocationHandler() {
     public Object invoke(Object proxy, Method method,Object[] args) throws Throwable {
                    return null;
     }
    });// 这种方式 同第一种类似,只不过是将参数换成实现InvocationHandler接口的子类,而且是内部类
第三种:
final ArrayList target=new ArrayList();
Collection proxy3 = (Collection) Proxy.newProxyInstance(
            Collection.class.getClassLoader(),
            new Class[] { Collection.class },
    new InvocationHandler() {
     public Object invoke(Object proxy, Method method,
       Object[] args) throws Throwable {
       retObj = method.invoke(target, args); 
       return retObj;     
           }
    });
proxy3.add("abcd");
  System.out.println(proxy3);
//这种方法即使用Proxy类中一个静态方法newProxyInstance(ClassLoader loader.class<?>[] interfaces,InvocationHandler h)创建。
注:当执行proxy3.add("abcd");时,代理对象proxy3对应invoke方法中proxy参数;.add方法对应invoke方法中method参数;而"abcd"参数则对应invoke方法中args参数,InvocationHandler接口的执行原理就是如此。
如果把目标对象和在代理类中添加的系统功能封装成对象传入代理类,即面向方面编程:把切面的代码以对象的方式进行封装,传给代理类。
例如:
public static Object getProxy(final Object target, final Advice advice) {
  Object proxy3 = Proxy.newProxyInstance(
    target.getClass() .getClassLoader(),
    target.getClass().getInterfaces(),
    new InvocationHandler() {
     @Override
     public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            advice.beforeExecute(method);
            Object retVal = method.invoke(target, args);
             advice.afterExecute(method);
             return retVal;
           }
   });
  return proxy3;
}
//advice是一个接口,它有两个方法,在它的实现类中实现在代理类中其它系统功能,即把系统功能封装成advice对象,以参数形式传入获得代理类的方法中。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: 
相关文章推荐