动态代理

2018-08-10  本文已影响0人  虫儿飞ZLEI

layout: post
title: 动态代理
subtitle: 用法
date: 2018-05-07
author: ZL
header-img: img/20180507.jpg
catalog: true
tags:
- 动态代理


动态代理

反射的原理,在运行时执行某个方法。

和反射不一样的是:
反射只能执行方法,而动态代理不仅可以执行方法,还可以做一些修改,比如改方法的返回值,改方法的参数等等。

例子

首先有一个接口:

public interface TargetInterface {

    public void method1();
    public String method2();
    public int method3(int x);
}

然后有一个接口的实现类


public class Target implements TargetInterface{

    @Override
    public void method1() {
        System.out.println("method1 running...");
    }

    @Override
    public String method2() {
        System.out.println("method2 running...");
        return "method2";
    }

    @Override
    public int method3(int x) {
        return x;
    }
}

最后动态代理执行方法


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

import org.junit.Test;

public class ProxyTest {

    @Test
    public void test1(){
        //获得动态的代理对象----在运行时 在内存中动态的为Target创建一个虚拟的代理对象
        //objProxy是代理对象 根据参数确定到底是谁的代理对象
        TargetInterface objProxy = (TargetInterface) Proxy.newProxyInstance(
                Target.class.getClassLoader(), //与目标对象相同的类加载器
                new Class[]{TargetInterface.class},
                new InvocationHandler() {
                    //invoke 代表的是执行代理对象的方法
                    @Override
                    //method:代表目标对象的方法字节码对象
                    //args:代表目标对象的响应的方法的参数
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
            //在方法执行前可以做需要的修改
                        System.out.println("目标方法前的逻辑");
                        //执行目标对象的方法
                        Object invoke = method.invoke(new Target(), args);
            //在方法执行后也可以做修改,比如修改返回值
                        System.out.println("目标方法后的逻辑");

            //这里返回的就是Target方法执行后的返回值,如果原方法没有返回值,就返回null
                        return invoke;
                    }
                }
            );

        objProxy.method1();
        String method2 = objProxy.method2();
        System.out.println(method2);

    }

}

上面动态代理的写法还可以换个方式写(一点点差别)

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

public class ProxyTest2 {

    public static void main(String[] args) {

        final Target target = new Target();

        //动态创建代理对象

        TargetInterface proxy = (TargetInterface) Proxy.newProxyInstance(
                target.getClass().getClassLoader(),
                target.getClass().getInterfaces(),
                new InvocationHandler() {
                    @Override
                    //被执行几次?------- 看代理对象调用方法几次
                    //代理对象调用接口相应方法 都是调用invoke
                    /*
                     * proxy:是代理对象
                     * method:代表的是目标方法的字节码对象
                     * args:代表是调用目标方法时参数
                     */
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        //反射知识点
                        Object invoke = method.invoke(target, args);//目标对象的相应方法
                        //retrun返回的值给代理对象
                        return invoke;
                    }
                }
            );

        proxy.method1();//调用invoke---Method:目标对象的method1方法  args:null  返回值null
        String method2 = proxy.method2();//调用invoke---Method:目标对象的method2方法  args:null  返回值method2
        int method3 = proxy.method3(100);////调用invoke-----Method:目标对象的method3方法 args:Object[]{100}  返回值100

        System.out.println(method2);
        System.out.println(method3);

    }

}

上一篇 下一篇

猜你喜欢

热点阅读