AOP-代理模式

一、代理模式?

1.1 概念

①介绍二十三种设计模式中的一种,属于结构型模式。它的作用就是通过提供一个代理类,让我们在调用目标
方法的时候,不再是直接对目标方法进行调用,而是通过代理类间接调用。让不属于目标方法核心逻辑
的代码从目标方法中剥离出来——解耦。调用目标方法时先调用代理对象的方法,减少对目标方法的调
用和打扰,同时让附加功能能够集中在一起也有利于统一维护。
在这里插入图片描述
使用代理后:
在这里插入图片描述
②生活中的代理

  • 广告商找大明星拍广告需要经过经纪人
  • 合作伙伴找大老板谈合作要约见面时间需要经过秘书
  • 房产中介是买卖双方的代理

③相关术语
代理:将非核心逻辑剥离出来以后,封装这些非核心逻辑的类、对象、方法。
目标:被代理“套用”了非核心逻辑代码的类、对象、方法。

1.2 静态代理

创建静态代理类:

package com.atguigu.spring.proxy;

public class CalculatorStaticProxy implements Calculator{

    private CalculatorImpl target;

    public CalculatorStaticProxy(CalculatorImpl target) {
        this.target = target;
    }

    public int add(int i, int j) {
        int result = 0;
        try {
            System.out.println("日志,方法:add,参数:"+i+","+j);
            result = target.add(i,j);
            System.out.println("日志,方法:add,结果:"+result);
        } catch (Exception e) {
            e.printStackTrace();
        } finally {
        }
        return result;
    }

    public int sub(int i, int j) {
        System.out.println("日志,方法:sub,参数:"+i+","+j);
        int result = target.sub(i,j);
        System.out.println("日志,方法:sub,结果:"+result);
        return result;
    }

    public int mul(int i, int j) {
        System.out.println("日志,方法:mul,参数:"+i+","+j);
        int result = target.mul(i,j);
        System.out.println("日志,方法:mul,结果:"+result);
        return result;
    }

    public int div(int i, int j) {
        System.out.println("日志,方法:div,参数:"+i+","+j);
        int result = target.div(i,j);
        System.out.println("日志,方法:div,结果:"+result);
        return result;
    }
}

静态代理确实实现了解耦,但是由于代码都写死了,完全不具备任何的灵活性。就拿日志功能来
说,将来其他地方也需要附加日志,那还得再声明更多个静态代理类,那就产生了大量重复的代
码,日志功能还是分散的,没有统一管理。
提出进一步的需求:将日志功能集中到一个代理类中,将来有任何日志需求,都通过这一个代理
类来实现。这就需要使用动态代理技术了。

1.3 动态代理

在这里插入图片描述
生产代理对象的工厂类:

public class ProxyFactory {
	private Object target;
	public ProxyFactory(Object target) {
	this.target = target;
	}
	public Object getProxy(){
	/**
	* newProxyInstance():创建一个代理实例
	* 其中有三个参数:
	* 1、classLoader:加载动态生成的代理类的类加载器
	* 2、interfaces:目标对象实现的所有接口的class对象所组成的数组
	* 3、invocationHandler:设置代理对象实现目标对象方法的过程,即代理类中如何重写接
	口中的抽象方法
	*/
	ClassLoader classLoader = target.getClass().getClassLoader();
	Class<?>[] interfaces = target.getClass().getInterfaces();
	InvocationHandler invocationHandler = new InvocationHandler() {
	@Override
	public Object invoke(Object proxy, Method method, Object[] args)
	throws Throwable {
	/**
	* proxy:代理对象
	* method:代理对象需要实现的方法,即其中需要重写的方法
	* args:method所对应方法的参数
	*/
	Object result = null;
	try {
		System.out.println("日志,方法:"+method.getName()+",参
		数:"+ Arrays.toString(args));
		result = method.invoke(target, args);
		System.out.println("日志,方法:"+method.getName()+",结
		果:"+ result);
	} catch (Exception e) {
		e.printStackTrace();
		System.out.println("日志,方法:"+method.getName()+",异
		常:"+e.getMessage());
	} finally {
		System.out.println("日志,方法:"+method.getName()+",方法
		执行完毕");
	}
	return result;
	}
};
	return Proxy.newProxyInstance(classLoader, interfaces, invocationHandler);
	}
}

测试:

public class proxyTest {
    /**
     * 动态代理有两种:
     * 1、jdk动态代理,要求必须有接口,最终生成的代理类和目标类实现相同的接口
     * 在com.sun.proxy包下,类名为$proxy2
     * 2、cglib动态代理,最终生成的代理类会继承目标类,并且和目标类在相同的包下
     */

    @Test
    public void testProxy(){
        /*CalculatorStaticProxy proxy = new CalculatorStaticProxy(new CalculatorImpl());
        proxy.add(1,2);*/
        ProxyFactory proxyFactory = new ProxyFactory(new CalculatorImpl());
        Calculator proxy = (Calculator) proxyFactory.getProxy();
        proxy.add(1,2);
        //proxy.div(1,0);
    }
}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值