开发者学堂课程【Java Web 开发系列课程:Spring 框架入门:09代理的讲解】学习笔记,与课程紧密联系,让用户快速学习知识。
课程地址:https://developer.aliyun.com/learning/course/538/detail/7342
动态代理
内容简介:
一、动态代理介绍
二、Javasstst 层
三、方法详细信息
四、Proxy 方法
一、动态代理介绍
1、动态代理和静态代理的角色是一样的。
2、动态代理的代理类是动态生成的。
3、分为两类一类基于 接口动态代理和基于类的动态代理
a)基于接口动态代理 -idk 动态代理
b)基于类的动态代理 -cglib
现在 javasist 来生成动态代理
4、jdk 动态代理—— Proxy 类和 InvocationHandler 接口
InvocationHandler 是代理实例的调用处理程序实现的接口。
每个代理实例都具有一一个关联的调用处理程序。对代理实例调用方法时,将对方法调用进行编码并将其指派到它的调用处理程序的invoke 方法。
二、 Javasstst 层
Javasstst 层一个开源的分析、编销和创建 Java 字节码的英库。是由东京工业大学的教学和计算机科学系的 Shigeru Cniba (千叶)所创建的。
它已加入了开放源代码 oss 应用服务器项目通过使用Javassst 对字节码操作为 JBoss 实现动态 AOP 架。
关于 java 字节码的处理,目前有供多工具,如 bce , asm 。不过这些都需要自接跟告报机指令打交道。
如果你不想了解感扣机指令,可以采用 awassstjavassist 是 boss 的一个子项目,其主要的优点,在于简单,而且快速。真接使用 java编的的形式,而不需要了解虎拟机指令,就能动态改变类的结构,或者动态生成类。
三、方法详细信息
invoke
Object invoke(object proxy,Method method Object[] args) throws Throwable
在代理实例上处理方法调用并返回结果。在与方法关联的代理实例上调用方法时,将在调用处理程序上调用此方法。
参数:
proxy- 在其上调用方法的代理实例。 method- 对应于在代理实例上调用的接口方法的 Method 实例。 Method 对象的声明类将是在其中声明方法的接口,该接口可以是代理类赖以继承方法的代理接口的超接口。
args- 包含传入代理实例上方法调用的参数值的对象数组,如果接口方法不使用参数,则为 null 。基本类型的参数被包装在适当基本包装器类(如 java.lang.Integer 或 java.lang.Boolean )的实例中。
返回:
从代理实例的方法调用返回的值。如果接口方法的声明返回类型是基本类型,则此方法返回的值一定是相应基本包装对象类的实例:否则,它一定是可分配到声明返回类型的类型。如果此方法返回的值为nul1 并且接口方法的返回类型是基本类型,则代理实例上的方法调用将抛出 NullPointerException 。否则,如果此方法返回的值与上述接口方法的声明返回类型不兼容,则代理实例上的方法调用将抛出ClassCastException 。
抛出:
Throwable- 从代理实例上的方法调用抛出的异常。该异常的类型必须可以分配到在接口方法的 throws 子句中声明的任一异常类型或未经检查的异常类型 java.lang.RuntimeException 或 java.lang.Error 。如果此方法抛出经过检查的异常,该异常不可分配到在接口方法的 throws 子句中声明的任一异常类型,代理实例的方法调用将抛出包含此方法曾抛出的异常的 UndeclaredThrowableException
另请参见:
UndeclaredThrowableException
四、Proxy 方法
Proxy 提供用于创建动态代理类和实例的静态方法,它还是由这些方法创建的所有动态代理类的超类。
Static Object
newProxyInstance(ClassLoader loader,
Class[ ] interfaces, InvocationHandler h)
返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。
返回一个指定接口的代理类实例,该接口可以将方法调用指派到指定的调用处理程序。此方法相当于:
Proxy.getProxyClass(loader, interfaces).
getConstructor(new Class[ ] {InvocationHandler.class}).
newInstance(new Object[ ] { handler });
Proxy.newProxyInstanc抛出IllegalArgumentException,原因与 Proxy.getProxyClass相同。
参数:
loader- 定义代理类的类加载器
interfaces- 优理类要实现的接口列表
h- 指派方法调用的调用处理程序
代码如下:
public class ProxyInovationHandler implements
InvocationHandler{
private Object target;//目标对象--真实对象
public void setTarget(object target){
this.target = target;
}
/**
public object getproxyo(){
return
Proxy.newProxyinstance(this.hetClass(),getclassloader(),
target,getclass() getintertaces(),this);
/**
*Proxy是代理类
*method 代理类的调用处理程序的方法对象
**/
@overide
public object invoke(object proxy, method method,object [ ]
args)
throws Throwable(
log(method.getName());
Object result= method.invoke(target, args);
return result;
}
public void log(string methodtame){
throws Throwable
{ log(method.getName());
Object result=methodinvoke(target,args) ;
return result;
}
public void log(string methodName){
System.out.println("执行"+methodName+"方法
");
}
}
Chent.java 客户
public class client {
public static void main(string[ ] args){
Host host =new Host();
Proxy proxy=new Proxy(host);
proxy.rent();
}
}