Java学习 动态代理

什么是动态代理?

例子:
在这里插入图片描述
明星会唱歌,跳舞。但是开演唱会如果还兼顾其他事务工作的话,事情就会太多,因此这时候就可以使用代理,转移部分事情。
(代理会先把准备工作做完,然后再去调用对象中的方法)
⚪对象如果嫌身上干的事太多的话,可以通过代理来转移部分职责。
⚪对象有什么方法想被代理,代理就一定要有对应的方法.

1.为什么需要代理?

代理可以无侵入式的给对象增强其他的功能
调用者–>代理–>对象

2.代理长什么样?

代理的里面就是对象要被代理的方法

3.Java通过上面来保证代理的样子?

通过接口保证,后面的对象和代理需要实现同一个接口
接口中就是被代理的所有方法

如何为Java对象创建一个代理对象

java.lang.reflect.Proxy类型:提供了为对象产生代理对象的方法:
public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)
参数一:用于指定用哪个类加载器,去加载生成的代理类
参数二:指定接口,这些接口用于指定生成的代理长什么,也就是有哪些方法。
参数三:用来指定生成的代理对象要干上面事情。

案例:

明星类:BigStar

package Xia;

public class BigStar implements Star {
    private String name;

    public BigStar() {
    }

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

    //唱歌,跳舞
    public String sing(String geming) {
        System.out.println(this.name + "正在唱" + geming);
        return "谢谢";
    }

    public void dance() {
        System.out.println(this.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 + "}";
    }
}

接口:Star 里面是想要被代理的所有方法

package Xia;

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

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

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

代理类 Proxy :创建一个代理对象,重中之重

package Xia;

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<?>[] interfaces,InvocationHandler h)
          参数一:用于指定用哪个类加载器,去加载生成的代理类
          参数二:指定接口,这些接口用于指定生成的代理长什么,也就是有哪些方法。
          参数三:用来指定生成的代理对象要干上面事情。*/
        Star star = (Star) Proxy.newProxyInstance(
                //第一个获取这个类的class对象,然后class对象调用方法寻找这个类加载进行内存的加载器。   可以理解为::找个人给自己送进去
                ProxyUtil.class.getClassLoader(),
                //可以理解为: 中介可以代理哪些方法,如果有多个方法都可以写在数组中
                new Class[]{Star.class},       //   代理和代理对象都需要实现的接口方法
                //
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {   //代理实现方法是真正执行的流程
                        //参数一: 代理的对象
                        //参数二: 要运行的方法 sing
                        //参数三: 调用方法是要传递的 参数
                        if ("sing".equals(method.getName())) {
                            System.out.println("准备话筒,收钱");
                        }
                        if ("dance".equals(method.getName())) {
                            System.out.println("准备场地,收钱");
                        }
                        //去找大明星开始唱歌跳舞
                        //代码的表现形式:调用大明星里面唱歌或者跳舞的方法
                        return method.invoke(bigStar, args);
                    }
                });
        return star;
    }
}

实现代理

package Xia;

public class DaiLiTest {
    public static void main(String[] args) {
        /*
         * 需求:  外面的人想要大明星唱一首歌
         *  1,获取代理的对象
         *   代理对象 = ProxyUtil.createProxy(大明星对象);
         *  2.再调用代理的唱歌方法
         *   代理对象.唱歌的方法(”及你太美“);
         */
        //1 . 获取代理对象
        BigStar bigStar = new BigStar("鸡哥");  //被代理的明星对象
        Star proxy = ProxyUtil.createProxy(bigStar);  // 给鸡哥生成了一个代理对象

        //调用唱歌的方法
        String result = proxy.sing("及你太美");  //  proxy会自动调用invoke方法,会将sing 和 及你太美 传入进去。
        System.out.println(result);

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

具体执行流程

在这里插入图片描述

1.通过代理实现方法proxy.sing(“只因你太美”),方法的底层就会自动去调用invoke方法,并将 sing 和 "只因你太美"作为参数传递进去。

2.进入方法体进行判断,从而执行什么流程

3.找到对象执行相关的方法

4. 返回对应的值

5.返回相应的值

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值