本文最后更新于:星期四, 六月 18日 2020, 9:01 上午

Proxy创建动态代理的两个方法:

//方法一
static Class<?> getProxyClass(ClassLoader loader, Class<?>...interfaces);
//方法二
static Object newProxyInstance(ClassLoader loader, Class<?> interfaces, InvocationHander hander);

方法一:创建一个动态代理类所对应的Class对象,该代理类将实现interfaces数组所指定的多个接口。第一个ClassLoader参数指定生成动态代理类的类加载器。
方法二:直接创建一个动态代理对象,该对象的实现类实现了interfaces数组指定的系列接口,执行代理对象的每一个方法时都会被替换成InvocationHandler接口(自己创建一个类实现该接口)的实例的invoke方法。(实际上方法一需要创建对象的话,也是要一个InvocationHandler对象的)。


InvoactionHandler的使用方法:

//接口
public interface Person {
    void walk();
    void sayHello(String name);
}
public class MyInvocationHandler implements InvocationHandler{

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("---正在执行的方法:"+method);
        if (args!=null){
            System.out.println("下面是执行该方法时传入的实参为:");
            for (Object val:args){
                System.out.println(val);
            }
        }else System.out.println("该方法没有实参");
        return null;
    }
}
public class MainTest {
    public static void main(String args[]) throws Exception {
        InvocationHandler handler=new MyInvocationHandler();
        //注意这里的Person是一个接口interface。
        Person p= (Person) Proxy.newProxyInstance(Person.class.getClassLoader(),new Class[]{Person.class},handler);
        p.walk();
        p.sayHello("孙悟空");
    }
}

AOP代理:
AOP代理可替代目标对象,AOP代理对象包含了目标对象的全部方法,但AOP代理中的方法与目标对象的方法存在差异:AOP代理里的方法可以在执行目标方法之前、之后插入一些通用处理。

例子:

  • 思路:创建Dog对象和他的GunDog实现类,创建InvocationHandler的实现类MyInvocationHandler, 在invoke方法中调用目标对象的方法,并且在方法前后加入自定义的Utils类的方法, 创建代理对象工厂类MyProxyFactory,用于生产代理对象。

    public interface Dog {
      void info();
      void run();
    }
    public class GunDog implements Dog {
      @Override
      public void info() {
          System.out.println("我是一只猎狗");
      }
    
      @Override
      public void run() {
          System.out.println("我奔跑迅捷");
      }
    }
    public class Util {
      public void method1(){
          System.out.println("----模拟第一个通用方法----");
      }
      public void method2(){
          System.out.println("----模拟第二个通用方法----");
      }
    }
    public class MyInvocationHandler implements InvocationHandler {
      //被代理的对象;
      private Object target;
    
      public void setTarget(Object object) {
          this.target = object;
      }
      @Override
      public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
          Util u = new Util();
          u.method1();
          //注意这里第一个参数不要放proxy进去,因为我们要为指定的target生成动态代理对象
          //放proxy是新生成的动态代理对象。
          Object result = method.invoke(target, args);
          u.method2();
          return result;
      }
    }
    public class MyProxyFactory {
      public static Object getProxy(Object target){
          //创建一个MyInvocationHandler对象
          MyInvocationHandler handler=new MyInvocationHandler();
          //设置指定的要生成动态代理的target对象
          handler.setTarget(target);
          //创建并返回一个动态代理对象。
          return Proxy.newProxyInstance(target.getClass().getClassLoader(),target.getClass().getInterfaces(),handler);
      }
    }
    public class MainTest {
      public static void main(String args[]) throws Exception {
          GunDog dog=new GunDog();
          Dog proxyDog= (Dog) MyProxyFactory.getProxy(dog);
          proxyDog.info();
          proxyDog.run();
      }
    }

    打印:
    ```

—-模拟第一个通用方法—-
我是一只猎狗
—-模拟第二个通用方法—-
—-模拟第一个通用方法—-
我奔跑迅捷
—-模拟第二个通用方法—-

```

###结论:动态代理模式让代码结构更加灵活,解耦。


Java     

本博客所有文章除特别声明外,均采用 CC BY-SA 3.0协议 。转载请注明出处!

通过反射查看泛型信息 上一篇
泛型类型转换的问题 下一篇