黑马程序员--Java基础加强--08内省、类加载器、动态代理

来源:互联网 发布:北师珠网络登录 编辑:程序博客网 时间:2024/05/12 03:40

----------------------Android培训Java培训java学习型技术博客、期待与您交流! ----------------------

内省
内省(IntroSpector):主要用于对JavaBean进行操作。

JavaBean是一种特殊的java类,主要用于传递数据信息。
这种java类方法主要用于访问私有的字段,且方法名符合某种命名规则。

PropertyDescriptor 描述 Java Bean 通过一对存储器方法导出的一个属性。
          PropertyDescriptor(String propertyName, Class<?> beanClass) 
          通过调用 getFoo 和 setFoo 存取方法,为符合标准 Java 约定的属性构造一个 PropertyDescriptor。
          Method getReadMethod()  获得应该用于读取属性值的方法。 
          Method getWriteMethod()  获得应该用于写入属性值的方法。 
例子演示:

import java.beans.PropertyDescriptor;import java.lang.reflect.*;class IntroSpectorDemo{public static void main(String[] args) throws Exception{Person p = new Person("lxh",24);//同过变量的名称和所在类的字节码获取PropertyDescriptor对象String propertyName1 = "name";PropertyDescriptor pd1 = new PropertyDescriptor(propertyName1,p.getClass());//然后使用该对象调用getReadMethod()方法来获取读属性的方法的反射Method methodGet = pd1.getReadMethod();//通过反射把该类对象传递给invoke方法来调用对应的方法Object retVal = methodGet.invoke(p);System.out.println(retVal);//同过变量的名称和所在类的字节码获取PropertyDescriptor对象String propertyName2 = "age";PropertyDescriptor pd2 = new PropertyDescriptor(propertyName2,p.getClass());//然后使用该对象调用getWriteMethod()方法来获取写入属性的方法的反射Method methodSet = pd2.getWriteMethod();//通过反射把该类对象和相应的参数传递给invoke方法来调用对应的方法methodSet.invoke(p,23);System.out.println(p.getAge());}}//创建一个类用于JavaBean测试class Person{private String name;private int age;Person(String name, int age){this.name = name;this.age = age;}public void setAge(int age){this.age = age;}public int getAge(){return age;}public void setName(String name){this.name = name;}public String getName(){return name;}}
类加载器
BootStrap:加载JRE/lib/rt.jar
|--ExtClassLoader:加载JRE/lib/ext/*.jar
|--AppClassLoader:加载CLASSPATH指定的所有.jar或目录。

委托机制
每个类加载器加载类时,又先委托给其上级类加载器先去加载。
当所有祖宗类加载器没有加载到类,回到发起者类加载器还是加载不到则会抛出ClassNotFoundException.

加载原理
首先当前线程的类加载器去加载线程中的第一个类。
如果类A中引用了B类,java虚拟机将使用加载类A的类加载器来加载类B
还可以直接调用ClassLoader.loadClass()方法来指定某个类加载器去加载某个类
ClassLoader(抽象类)
自定义类加载器必须继承ClassLoader类
例子演示:

class ClassLoaderTest {public static void main(String[] args) {//获取加载当前类的类加载器System.out.println(ClassLoaderTest.class.getClassLoader().getClass().getName());//sun.misc.Launcher$AppClassLoader//获取加载System类的类加载器System.out.println(System.class.getClassLoader());//null//类加载器结构ClassLoader loader = ClassLoaderTest.class.getClassLoader();while(loader!=null){System.out.println(loader.getClass().getName());loader = loader.getParent();}System.out.println(loader);/*sun.misc.Launcher$AppClassLoadersun.misc.Launcher$ExtClassLoadernull*/}}
代理
代理是常用的java设计模式,它的特征是代理类与委托类有同样的接口,、
代理类主要负责为委托类预处理消息、过滤消息、把消息转发给委托类,以及事后处理消息等。

动态代理:在程序运行时,运用反射机制动态创建而成。

Proxy类 :java.lang.reflect包下
构造函数:
protected  Proxy(InvocationHandler h) 
          使用其调用处理程序的指定值从子类(通常为动态代理类)构建新的 Proxy 实例
方法:
static InvocationHandler getInvocationHandler(Object proxy) 
          返回指定代理实例的调用处理程序。 
static Class<?> getProxyClass(ClassLoader loader, Class<?>... interfaces) 
          返回代理类的 java.lang.Class 对象,并向其提供类加载器和接口数组。 
static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) 
          返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。 

例子演示:

import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;import java.util.ArrayList;import java.util.List;class ProxyTest{public static void main(String[] args){//测试List arrayListProxy = ArrayListProxy.getArrayListProxy();//添加元素arrayListProxy.add("abd");arrayListProxy.add("abd");arrayListProxy.add("abd");//获取元素个数System.out.println(arrayListProxy.size());//打印集合System.out.println(arrayListProxy);}}//定义一个ArrayList代理类class ArrayListProxy implements List{//定义一个静态方法来获取ArrayList的动态代理public static List getArrayListProxy(){//使用ArrayList的父类接口List来创建ArrayList的动态代理对象List arrayListProxy = (List)Proxy.newProxyInstance(List.class.getClassLoader(),new Class[]{List.class},new InvocationHandler(){//创建一个ArrayList对象private ArrayList target  = new ArrayList();@Overridepublic Object invoke(Object proxy, Method method,Object[] args)throws Throwable {//方法开始时间long startTime = System.currentTimeMillis();//执行方法Object retVal = method.invoke(target, args);//方法结束时间long endTime = System.currentTimeMillis();System.out.println(method.getName()+"()"+" running time : "+(endTime - startTime)+"毫秒");return retVal;}});return arrayListProxy;}}





----------------------Android培训Java培训java学习型技术博客、期待与您交流! ----------------------

详细请查看:http://edu.csdn.net/


0 0