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

----------------------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$AppClassLoader
		sun.misc.Launcher$ExtClassLoader
		null
		*/
	}
}
代理
代理是常用的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();
					@Override
					public 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/


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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值