Java:动态代理

动态代理的特点是无侵入式的给代码增加额外的功能。

调用者---->代理---->对象

java 的动态代理主要用于在运行时动态创建代理对象,从而拦截方法调用,实现 AOP(面向切面编程)、权限控制、日志记录等功能。Java通过接口来保证代理,后面的对象和代理需要实现同一个接口,接口中就是被代理的所有方法

创建代理对象

java.lang.reflect.Proxy类:提供了为对象产生代理对象的方法:

public static Object newProxyInstance (ClassLoader loader, class[ ]  interfaces, InvocationHandler h)
参数一:用于指定用哪个类加载器,去加载生成的代理类
参数二:指定接口,这些接口用于指定生成的代理长什么,也就是有哪些方法                                参数三:用来指定生成的代理对象要干什么事情

例子代码

package Myreflect;

public class BigStar implements Star {
    String name;

    public BigStar() {
    }

    public BigStar(String name) {
        this.name = name;
    }

    @Override
    public String sing(String name) {
        System.out.println(this.name + "在唱" + name);
        return "谢谢";
    }

    @Override
    public void dance() {
        System.out.println(name + "在跳舞");
    }

    /**
     * 获取
     *
     * @return name
     */
    public String getName() {
        return name;
    }

    /**
     * 设置
     *
     * @param name
     */
    public void setName(String name) {
        this.name = name;
    }

    public String toString() {
        return "BigStar{name = " + name + "}";
    }
}
package Myreflect;

public interface Star {
    public abstract String sing(String name);

    public abstract void dance();
}
package Myreflect;

import cn.hutool.aop.ProxyUtil;

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

public class proxyUtil {
    public static Star createProxy(BigStar bigStar) {
        Star star1 = (Star) Proxy.newProxyInstance(
                ProxyUtil.class.getClassLoader(),//参数一:用于指定用哪个类加载器,去加载生成的代理类
                new Class[]{Star.class},        //参数二:指定接口,这些接口用于指定生成的代理长什么,也就是有哪些方法
                new InvocationHandler() {        //参数三:用来指定生成的代理对象要干什么事情
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        if ("sing".equals(method.getName()))
                            System.out.println("(唱歌)准备话筒");
                        else if ("dance".equals(method.getName())) {
                            System.out.println("(跳舞)准备场地");
                        }
                        return method.invoke(bigStar, args);
                    }
                }
        );
        return star1;
    }
}
package Myreflect;

public class Test {
    /*
        外面的人想要大明星唱一首歌
        1.获取代理的对象
            代理对象 = ProxyUti1.createProxy(大明星的对象);
        2.再调用代理的唱歌方法
            代理对象.唱歌的方法("只因你太美")
    * */
    public static void main(String[] args) {
        BigStar bigStar = new BigStar("702");
        Star proxy = proxyUtil.createProxy(bigStar);
        String res = proxy.sing("半句再见");
        System.out.println(res);
        proxy.dance();
    }

}

proxy.singh和proxy.dance会自动调用invoke方法,传参过去                                                method.invoke(bigStar, args)会调用重写过的BigStar里的方法,返回值逐层返回

你可能感兴趣的:(Java基础,java,开发语言)