动态代理的简单实现

对于动态代理的理解

以买房这一事件为例,一般都不会是开发商直接向买家售卖房子。
原因有很多,例如,房子过多,处理不过来;仅有两方面,没有见证者,容易出现谈不拢情况等等。
在这时候,房产中介就应运而生了,可以由房产商将自己手底下的房子的相关信息告诉中介,再由中介帮其推销到市场,买家通过中介了解房子信息。这就是代理的概念由来。
但在实际中,在发生了买卖之后,房产商会修改其房源信息或者增加房子信息,导致买家接收到的信息有差别,或者中介需要不断更改其代理房子的相关信息。如此,静态代理显得弹性不足,依赖性过强。便产生了动态代理的理念。在不修改原来的信息的情况下,可以通过动态代理来实现对其信息的拓展。

以下是我们老师布置的例题,下面通过例题的解决来初步学习理解动态代理的原理。

在不修改Division类的定义前提下,请用动态代理的方式编写一个代理程序对Division的功能进行扩展修改,完成扩展后的除法计算功能。
要求
a、编写一个动态代理处理器类,扩展除法功能如下:
(1).在除法计算前,可以检验除数是否为0,如果为0,则计算结果为-9999,并输出错误提示:除零错误!
(2).在除法计算后,可以检查是否有余数,如果有余数,则输出余数提示:余数为xxx
b、编写一个测试类,从键盘输入两个数,生成动态代理对象,通过代理对象进行除法计算,并输出最终结果。
c、编写其他可能需要的接口或类文件。
注意:代理方法有返回值
public class Division {
public int divide(int a, int b){
int result=-1;
try{
System.out.println(“开始计算…”);
result=a/b;
System.out.println(“计算完毕…”);
return result;
}catch(Exception e){ }
return result;
}
}

题目给的 Division.java

public class Division {
    public int divide(int a,int b){
        int result = -1;
        try{
            System.out.println("开始计算====\n");
            result=a/b;
            System.out.println("计算完毕====\n");
            return result;
        }catch(Exception e){
            return result;
        }
    }
}

计算接口类 Calculate.java

/**
 * 计算接口类
 */
public interface Calculate {
    public int divisionBefore(int a,int b);
    public String divisionAfter(int a,int b);
}

Calculate接口的实现类 RealCalculate.java

/**
 * Calculate接口的实现类(对Division的扩展)
 */
public class RealCalculate implements Calculate{
    //创建一个Division对象
    Division division=new Division();
    @Override
    public int divisionBefore(int a, int b) {
        if(b==0) {
            System.out.println("除零错误!\n");
            return -9999;
        }else{
            return division.divide(a,b);
        }
    }

    @Override
    public String divisionAfter(int a, int b) {
        try{
            int c=a%b;
            if(c!=0) return ("余数为"+c+"\n");
            else return ("余数为0\n");
        }catch(Exception e){
            return "";
        }
    }
}

Calculate的动态代理类 DynamicProxy.java

/**
 * Calculate的动态代理类
 */
public class DynamicProxy implements InvocationHandler {
    //代理的真实对象
    private Object calculate;
    //给要代理的对象赋值
    public DynamicProxy(Object calculate){
        this.calculate=calculate;
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        //代理对象前的提示
        System.out.println("===Before agent===\n");
        //当代理对象调用真实对象的方法时,其会自动跳转到代理对象关联的handler对象的invoke方法进行调用并获取其返回值
        Object object=method.invoke(calculate, args);
        //代理对象后的提示
        System.out.println("===After agent===\n");
        return object;
    }
}

Calculate的动态代理测试 AgentTest.java

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Proxy;
import java.util.Scanner;

public class AgentTest {
    public static void main(String[] args) {
        //代理的对象
        Calculate realCalculate=new RealCalculate();
//        Calculate realCalculate=new Real();//无返回值动态代理测试
        //将对象传进InvocationHandler
        InvocationHandler handler = new DynamicProxy(realCalculate);

        /*
         * 通过Proxy的newProxyInstance方法来创建代理对象,其有三个参数
         * 第一个参数 handler.getClass().getClassLoader() ,这里使用handler这个类的ClassLoader对象来加载代理对象
         * 第二个参数realSubject.getClass().getInterfaces(),这里为代理对象提供的接口是真实对象所实行的接口,表示要代理的是该真实对象,这样就能调用这组接口中的方法了
         * 第三个参数handler,这里将这个代理对象关联到了上方的 InvocationHandler 对象
         */
        Calculate calculate= (Calculate) Proxy.newProxyInstance(handler.getClass().getClassLoader(),
                realCalculate.getClass().getInterfaces(),handler);

        Scanner scanner = new Scanner(System.in);
        System.out.print("请输入被除数a:");
        int a=scanner.nextInt();
        System.out.print("请输入除数b:");
        int b=scanner.nextInt();
//        int a=101;
//        int b=15;
        int resultBefore = calculate.divisionBefore(a,b);
        String resultAfter=calculate.divisionAfter(a,b);
        System.out.println(a+"/"+b+"="+resultBefore);
        if(!resultAfter.equals("")){
            System.out.println(a+"/"+b+"的"+resultAfter);
        }

    }
}

这样就实现了功能的扩展了。
这里使用的是JDK的动态代理,有其不足之处:JDK动态代理的代理类字节码在创建时,需要实现业务实现类所实现的接口作为参数。如果业务实现类是没有实现接口而是直接定义业务方法的话,就无法使用JDK动态代理了。(JDK动态代理重要特点是代理接口)并且,如果业务实现类中新增了接口中没有的方法,这些方法是无法被代理的(因为无法被调用)。动态代理只能对接口产生代理,不能对类产生代理。

在此之前要对反射的概念有一定的了解,能够更快地掌握动态代理的实现原理。

本次学习笔记就到这啦!愿与诸君共同学习!

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值