动态代理

一、什么是动态代理?
image
现在要给eat方法增加其它功能,例如吃饭之前添加拿筷子,盛饭。
在已有的代码中插入,直接修改代码,我们叫做侵入式修改。而在一个成熟的项目中,这样做是很危险的,可能全崩啦!
image
此时想要增加额外的功能而又不能修改原有代码,如何去做呢?
此时我们可以找一个代理先帮我们做拿筷子和盛饭两个准备工作,等真正吃饭时,再去调用Student里的eat方法吃饭。这其实就是动态代理。
image

二、程序为什么需要代理?代理长什么样?
image

image

三、如何为Java创建一个代理对象
java.long.reflect.Proxy类,提供了为对象产生代理对象的方法
image

package com.mydynamicproxy1;

public class Test {
    public static void main(String[] args) {
        /*需求:外面的人想要大明星唱一首歌
                * 1、获取代理的对象
                *   代理对象 = ProxyUtil.createProxy(大明星的对象)
                * 2.再调用代理的唱歌方法
                *   代理对象.唱歌的方法("只因你太美");
         */


        //1、获取代理的对象
        BigStar bigstar = new BigStar("鸡哥");
        Star proxy = ProxyUtil.createProxy(bigstar);
        //2.再调用代理的唱歌方法
        String result = proxy.sing("只因你太美");
        System.out.println(result);

        //3.调用跳舞的方法
        proxy.dance();
    }
}

package com.mydynamicproxy1;

public class BigStar implements Star {
    private String name;

    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(this.name + "正在跳舞");
    }

    public String getName() {
        return name;
    }
    public void setName(String name) {
        this.name = name;
    }

}

package com.mydynamicproxy1;

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

public class ProxyUtil {

    /*
    * 方法的作用:
    *   给一个明星的对象,创建一个代理
    *   形参:
    *       被代理的明星对象
    *   返回值:
    *       给明星创建的代理

    * 需求:
    *   外面的人想要大明星唱一首歌
    * 1、获取代理的对象
    *   代理对象 = ProxyUtil.createProxy(大明星的对象)
    * 2.再调用代理的唱歌方法
    *   代理对象.唱歌的方法("只因你太美");
    *
    * */

    public static Star createProxy(BigStar bigstar){
        /*java.lang.reflect.Proxy类:提供了为对象产生代理对象的方法:
        * public static Object newProxyInstance(ClassLoader loader,class<?>[] interface,InvocationHandler h)
        * 参数一:用于指定用哪个类加载器,去加载生成的代理类
        * 参数二:指定接口,这些接口用于指定生成的代理长什么样,也就是有哪些方法
        * 参数三:用来指定生成的代理对象要干什么事情
        * */
        Star star = (Star)Proxy.newProxyInstance(
                ProxyUtil.class.getClassLoader(),//参数一:用于指定用哪个类加载器,去加载生成的代理类
                new Class[]{Star.class},//参数二:指定接口,这些接口用于指定生成的代理长什么样,也就是有哪些方法
                //参数三:用来指定生成的代理对象要干什么事情
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
                        /*
                        * 参数一:代理的对象
                        * 参数二:要运行的方法 sing
                        * 参数三:调用sing方法时,传递的实参
                        *
                        * */
                        if("sing".equals(method.getName())){//判断要运行的方法是否在代理方法列表里(我代理的人可以干这个活,我先收费)
                            System.out.println("准备话筒,收钱");
                        }else if("dance".equals(method.getName())){
                            System.out.println("准备场地,收钱");
                        }
                        //去找大明星开始唱歌或跳舞
                        //代码的表现形式:调用大明星里面唱歌或者跳舞的方法
                        return method.invoke(bigstar,args);
                    }
                }
        );
        return star;
    }
}

package com.mydynamicproxy1;

public interface Star {
    //我们可以把所有想要被代理的方法定义在接口当中

    //唱歌
    public abstract String sing(String name);

    //跳舞
    public abstract void dance();
}

接下来要理清调用过程(个人理解梳理)
1.首先创建被代理对象的代理对象
2.使用创建的代理对象调用方法时底层会调用invoke,将要执行的方法传给invoke的第二个参数method,要执行的方法的参数传给invoke的第三个参数args。
3.在invoke中,判断传入的要运行的方法是否在代理列表内,若是,对其进行方法的加强,如例,为大明星唱歌做准备工作。
4.接下来,调用被代理对象的方法,method就是实际的方法,args就是实际方法的形参,此时实际方法就被调用了,执行实际方法,将实际方法的返回值返回。

image
image
image

posted @ 2025-01-22 23:19  吃俺老孙一棒槌  阅读(15)  评论(0)    收藏  举报