黑马程序员_(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对象,以参数形式传入获得代理类的方法中。
类加载器的委托机制
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对象,以参数形式传入获得代理类的方法中。
相关文章推荐
- Java基础---Java---基础加强---类加载器、委托机制、AOP、 动态代理技术、让动态生成的类成为目标类的代理、实现Spring可配置的AOP框架
- Java基础---Java---基础加强---类加载器、委托机制、AOP、 动态代理技术、让动态生成的类成为目标类的代理、实现Spring可配置的AOP框架
- Java中三种代理方式—— 静态代理与两种动态代理的实现机制
- Spring事务管理机制的实现原理-动态代理
- 利用匿名委托和动态事件注册机制实现应用程序状态监视
- JAVA的动态代理机制及Spring的实现方式
- java中的动态代理机制实现AOP
- AOP动态代理的实现机制
- 黑马程序员--Java学习加强23--动态代理机制
- Proxy动态代理机制的Java实现
- PHP 反射机制实现动态代理的代码
- PHP实现依赖注入-使用反射机制和动态代理技术 - 简单思想(咋个办呢 zgbn)
- 黑马程序员--09.动态与代理AOP--04【推理动态代理类内部结构I】【InvocationHandler实现类的具体写法】
- 详解Java动态代理的实现机制
- JAVA的动态代理机制及Spring的实现方式
- JAVA的动态代理机制及Spring的实现方式
- JDK动态代理实现机制
- [原创]JAVA的动态代理机制及Spring的实现方式
- 加强2注解。泛型。类加载器及其委托机制。代理的概念与作用原理,AOP概念。实现AOP功能的封装与配置。类似Spring。
- 转:Java 动态代理的内部实现机制(大体意思正确,写的还行的一篇文章)