Java动态代理(二)——jdk动态代理

一、什么是动态代理?
代理类在程序运行时创建的代理方式被成为动态代理。动态代理的代理类并不是在Java代码中定义的,而是在运行时根据我们在Java代码中的“指示”动态生成的。相比于静态代理, 动态代理的优势在于可以很方便的对代理类的函数进行统一的处理,而不用修改每个代理类中的方法。

Java动态代理有两种:jdk动态代理和Cglib动态代理

 

二、JDK动态代理
在java的动态代理机制中,有两个重要的类或接口,一个是 InvocationHandler(Interface)、另一个则是 Proxy(Class),这一个类和接口是实现动态代理所必须用到的东西。

 

1.InvocationHandler:
每一个动态代理类都必须要实现InvocationHandler这个接口,并且每个代理类的实例都关联到了一个handler,当我们通过代理对象调用一个方法的时候,这个方法的调用就会被转发为由InvocationHandler这个接口的 invoke 方法来进行调用。我们来看看InvocationHandler这个接口唯一的一个方法 invoke 方法:

Object invoke(Object proxy, Method method, Object[] args) throws Throwable

其接受三个参数,那么这三个参数分别代表什么呢?

  • proxy:  指代我们所代理的那个真实对象
  • method:  指代的是我们所要调用真实对象的某个方法的Method对象
  • args:  指代的是调用真实对象某个方法时接受的参数

 

2.Proxy:
Proxy这个类的作用就是用来动态创建一个代理对象的类,它提供了许多的方法,但是我们用的最多的就是 newProxyInstance 这个方法:

public static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h) throws IllegalArgumentException

其接收三个参数,我们来看看这三个参数所代表的含义:

  • loader:  一个ClassLoader对象,定义了由哪个ClassLoader对象来对生成的代理对象进行加载
  • interfaces: 一个Interface对象的数组,表示的是我将要给我需要代理的对象提供一组什么接口,如果我提供了一组接口给它,那么这个代理对象就宣称实现了该接口(多态),这样我就能调用这组接口中的方法了
  • h:  一个InvocationHandler对象,表示的是当我这个动态代理对象在调用方法的时候,会关联到哪一个InvocationHandler对象上。

3.用上一篇的Person例子来进一步运用
(1)定义一个IPerson接口

public interface IPerson {
public String getName();

public Integer getAge();
}

(2)定义一个真实的对象类来实现这个接口 

public class PersonImp implements IPerson {
private String name;
private Integer age;

public PersonImp() {
super();
}

public PersonImp(String name, Integer age) {
super();
this.name = name;
this.age = age;
}

@Override
public String getName() {
// TODO Auto-generated method stub
return name;
}

@Override
public Integer getAge() {
// TODO Auto-generated method stub
return age;
}
}

(3)定义一个动态代理类,实现InvocationHandler接口

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class MyInvocationHandler implements InvocationHandler {
private Object target;

public MyInvocationHandler() {
super();
}

public MyInvocationHandler(Object target) {
super();
this.target = target;
}

@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
if ("getName".equals(method.getName())) {
System.out.println("**********before*********");
System.out.println("method:" + method);
Object res = method.invoke(target, args);
System.out.println("***********after*********");
return res;
} else {
Object res = method.invoke(target, args);
return res;
}
}
}

(4)测试一下

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;

public class PersonProxyTest {
public static void main(String[] args) {
IPerson person = new PersonImp("jack", 20);//我们要代理的真实对象(被代理对象)
InvocationHandler handler = new MyInvocationHandler(person);//代理对象,代理哪个对象,就把哪个对象传进去

IPerson personProxy = (IPerson) Proxy.newProxyInstance(person.getClass().getClassLoader(),
person.getClass().getInterfaces(), handler);
System.out.println(personProxy.getClass().getName());
System.out.println(personProxy.getName());
System.out.println(personProxy.getAge());
}
}

(5)看一下输出结果:

 

我们看看 com.sun.proxy.$Proxy0 这个东西,这个东西是由 System.out.println(personProxy.getClass().getName()); 这条语句打印出来的,那么为什么我们返回的这个代理对象的类名是这样的呢?
原因是:通过 Proxy.newProxyInstance 创建的代理对象是在jvm运行时动态生成的一个对象,它并不是我们的InvocationHandler类型,也不是我们定义的那组接口的类型,而是在运行是动态生成的一个对象,并且命名方式都是这样的形式,以$开头,proxy为中,最后一个数字表示对象的标号。

posted @ 2019-02-26 19:02  橘子洲头。  阅读(213)  评论(0编辑  收藏  举报