博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
jdk动态代理
阅读量:7137 次
发布时间:2019-06-28

本文共 4033 字,大约阅读时间需要 13 分钟。

package com.itcast.day3;import java.io.ObjectInputStream.GetField;import java.lang.reflect.Constructor;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;import java.util.ArrayList;import java.util.Collection;/** * 通过反射打印出  代理类的 构造函数、方法以及其参数列表 * @author liujl * */public class ProxyTest {	public static void main(String[] args) throws Exception{		Class clazzProxy=Proxy.getProxyClass(Collection.class.getClassLoader(), Collection.class);		System.out.println(clazzProxy);		System.out.println("---------------begin constructors list-----------------");		Constructor[] constructors=clazzProxy.getConstructors();		for(Constructor constructor:constructors){			String name=constructor.getName();			StringBuilder sb=new StringBuilder(name);			sb.append("(");				Class[] clazzParams=constructor.getParameterTypes();				for(Class clazzParam:clazzParams){					sb.append(clazzParam.getName());				}			if(clazzParams!=null&&clazzParams.length!=0)				sb.deleteCharAt(sb.length()-1);			sb.append(")");			System.out.println(sb);		}					System.out.println("---------------begin methods list-----------------");		Method[] methods=clazzProxy.getMethods();		for(Method method:methods){ 			String name=method.getName();			StringBuilder sb=new StringBuilder(name);			sb.append("(");				Class[] clazzParams=method.getParameterTypes();				for(Class clazzParam:clazzParams){					sb.append(clazzParam.getName()).append(",");				}			if(clazzParams!=null&&clazzParams.length!=0)				sb.deleteCharAt(sb.length()-1);			sb.append(")");			System.out.println(sb);		}						System.out.println("---------------begin create instance object----------------");//		clazzProxy.newInstance();//这样不行,这样只会掉那个不带参数的构造方法,而代理对象没有无参构造						//第一种方式创建实例		Constructor constructor=clazzProxy.getConstructor(InvocationHandler.class);		class MyInvocationHander1 implements InvocationHandler{			@Override			public Object invoke(Object proxy, Method method, Object[] args)					throws Throwable {								return null;			}					}		Collection proxy1=(Collection) constructor.newInstance(new MyInvocationHander1());		System.out.println(proxy1.toString());		proxy1.clear();//		proxy1.size();//java.lang.NullPointerException						//第二种方式创建实例		Collection proxy2=(Collection)constructor.newInstance(new InvocationHandler(){			@Override			public Object invoke(Object proxy, Method method, Object[] args)					throws Throwable {								return null;			}		});												//第三种方式创建代理类的实例,  得到Class  和  创建实例对象  一步到位		Collection proxy3=(Collection)Proxy.newProxyInstance(Collection.class.getClassLoader(), 				new Class[]{Collection.class},				new InvocationHandler() {				ArrayList  target=new ArrayList();//类变量				@Override				public Object invoke(Object proxy, Method method, Object[] args)						throws Throwable {					StringBuilder sbMethodAndParams=new StringBuilder();					sbMethodAndParams.append(method.getName()).append("(");					if(args!=null){						for(Object obj : args){							sbMethodAndParams.append(obj.toString()).append(",");						}						if(args!=null&&args.length!=0){							sbMethodAndParams.deleteCharAt(sbMethodAndParams.length()-1);						}					}					sbMethodAndParams.append(")");					System.out.println(sbMethodAndParams);										long beginTime=System.currentTimeMillis();					Object retVal=method.invoke(target, args);					long endTime=System.currentTimeMillis();					System.out.println(method.getName()+"执行时间 "+(endTime-beginTime)+" 毫秒");					return retVal;				}		});				proxy3.clear();				proxy3.add("ljl");		proxy3.add("wiseq");		proxy3.add("traits");		System.out.println("集合元素的个数="+proxy3.size());				//Proxy也是肯定继承自Object  ,  		//proxy3.getClass()为啥不调用目标类的getClass()得到ArrayList的字节码? 		//那是因为Object只有三个方法委托给了InvocationHander,  分别是 toString 、hashCode 、 equals   ,而getClass()方法,生成的代理类有自己的实现		System.out.println(proxy3.getClass().getName());				}}

 

img_f1bc2e5f583ec27645ba35815aa5d5b7.jpg

把切面的方法以对象的方式封装,把对象传递给代理对象,代理对象执行传入的对象方法 就等于执行了切面的代码

 

 

想上面我们看到的硬编码方式的动态代理在实际的开发中是没有任何意义的,要变的更有用,需要将一些东西抽出去,实现参数化 代理

 

 

下一步: 将这个功能封装成一个通用的黑盒子,将目标和系统功能作为参数传入到代理对象中

最终抽取的样子,请看 

开始做,坚持做,重复做

转载地址:http://ictrl.baihongyu.com/

你可能感兴趣的文章
复习模块
查看>>
Codeforces 691C Exponential notation(模拟)
查看>>
spring-excel-导出
查看>>
利用jenkins打造通过自定义参数更新svn 指定文件任务
查看>>
PID三种参数的理解
查看>>
07.移动端类库
查看>>
nginx支持https
查看>>
target与currentTarget与this的区别
查看>>
查看进程信息
查看>>
转载:DIV+CSS有可能遇到的问题
查看>>
protocol buffer
查看>>
web常用模块测试用例
查看>>
【转】分布式数据层 TDDL 来自:阿里巴巴
查看>>
swing常用布局
查看>>
#学习笔记#e2e学习使用(二)
查看>>
LeetCode 222.完全二叉树的节点个数(C++)
查看>>
MyBatis(增删改查)
查看>>
.net编译指令
查看>>
java调用python的几种用法(看这篇就够了)
查看>>
20180307-Xen、KVM、VMware、hyper-v等虚拟化技术的比较
查看>>