java之反射篇(下)——动态代理

一、什么是动态代理

在工作中,有一个实现好的类,后面我要给这个类的方法增加一些功能,但是我们不能直接在源代码上面修改(侵入式修改),因为改错了,整个项目可能会瘫痪。于是我们需要一个代理类做额外的增加的功能,在对应的位置通过反射获取源代码对应的功能

简单说就是,我有一个类叫做鸡哥,它有唱和跳的功能,现在给鸡哥增加准备话筒,场地收钱的额外功能,但是鸡哥只想唱和跳,于是通过中介公司请了一个代理,代理需要做的就是准备话筒场地收钱,至于唱和跳就通过反射,获取鸡哥的成员方法来使用。

Java通过什么来保证代理的样子?

通过接口保证,后面的对象和代理需要实现同一个接口(就像上面的中介公司)
接口中就是被代理的所有方法。

二、实现实例代码

要增加功能的类的代码

要给鸡哥唱歌增加准备话筒和收钱的功能

要给鸡哥跳舞增加准备场地和收钱的功能

public class BigStar implements Star {
    private 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(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 + "}";
    }
}

代理接口 (指定被代理的所有方法)

就是要增加的功能的方法抽象出来

public interface Star {

    //中介公司
    //我们可以把所有想要被代理的方法定义在接口当中

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

    //跳舞
    public void dance();
}

代理工具类,用于建立一个代理

测试类就可以调用下面的代码创建一个代理,运行指定鸡哥方法时,先跑代理的功能代码。

这样鸡哥只用唱跳了

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(
                ProxyUtil.class.getClassLoader(),
                new Class[]{Star.class},
                new InvocationHandler() {
                    @Override
                    public Object invoke(Object proxy, Method method, Object[] args) throws InvocationTargetException, IllegalAccessException {
                        if ("dance".equals(method.getName())){
                            System.out.println("代理对象在唱歌前,准备场地,收钱");
                        }else if ("sing".equals(method.getName())){
                            System.out.println("代理对象在唱歌前,准备话筒,收钱");
                        }
                        return method.invoke(bigStar,args);
                    }
                }
        );
        return star;

    }
}

测试类

public class Test {
    public static void main(String[] args) {
        BigStar bigStar = new BigStar("鸡哥");
        Star star = ProxyUtil.createProxy(bigStar);
        String str = star.sing("只因太美");
        System.out.println(str);
        star.dance();
    }
}

输出结果

 

  • 6
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值