代理设计模式

介绍

其实每个模式名称就表明了该模式的作用,代理模式就是多一个代理类出来,替原对象进行一些操作。代理又分为动态代理和静态代理。

静态代理

比如我们在租房子的时候回去找中介,为什么呢?因为你对该地区房屋的信息掌握的不够全面,希望找一个更熟悉的人去帮你做,此处的代理就是这个意思。再如我们有的时候打官司,我们需要请律师,因为律师在法律方面有专长,可以替我们进行操作,表达我们的想法。先来看看关系图:
在这里插入图片描述
根据上文的阐述,代理模式就比较容易的理解了,我们看下代码:

public interface Sourceable {
 	public void method();
  }
public class Source implements Sourceable { 
	@Override 
	public void method() {
		 System.out.println("the original method!"); 
	 } 
}
public class Proxy implements Sourceable { 
	private Source source; public Proxy(){ 
		super(); 
		this.source = new Source(); 
}
		@Override 
		public void method() { 
		before(); 
		source.method(); 
		atfer(); 
}
	private void atfer() { 
		System.out.println("after proxy!"); 
}
	private void before() { 
		System.out.println("before proxy!");
	 } 
}

测试类:

public class ProxyTest { 
public static void main(String[] args) { 
		Sourceable source = new Proxy(); 
		source.method(); 
	}
}

输出:

before proxy! 
the original method!
after proxy!

代理模式的应用场景:
如果已有的方法在使用的时候需要对原有的方法进行改进,此时有两种办法:

  1. 修改原有的方法来适应。这样违反了“对扩展开放,对修改关闭”的原则。
  2. 就是采用一个代理类调用原有的方法,且对产生的结果进行控制。这种方法就是代理模式。
    使用代理模式,可以将功能划分的更加清晰,有助于后期维护!

动态代理

JDK动态代理
基于接口去实现的动态代理

public class JDKProxyFactory implements InvocationHandler {
    // 目标对象的引用 private Object target;
    // 通过构造方法将目标对象注入到代理对象中
    public JDKProxyFactory(Object target) {
        super();
        this.target = target;
    }

    /*** @return */
    public Object getProxy() {
        // 如何生成一个代理类呢?
        // 1、编写源文件
        // 2、编译源文件为class文件
        // 3、将class文件加载到JVM中(ClassLoader)
        // 4、将class文件对应的对象进行实例化(反射)
        // Proxy是JDK中的API类
        // 第一个参数:目标对象的类加载器
        // 第二个参数:目标对象的接口
        // 第二个参数:代理对象的执行处理器
        Object object = Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
        return object;
    }

    /*** 代理对象会执行的方法 */
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Method method2 = target.getClass().getMethod("saveUser", null);
        Method method3 = Class.forName("com.sun.proxy.$Proxy4").getMethod("saveUser", null);
        System.out.println("目标对象的方法:" + method2.toString());
        System.out.println("目标接口的方法:" + method.toString());
        System.out.println("代理对象的方法:" + method3.toString());
        System.out.println("这是jdk的代理方法");
        // 下面的代码,是反射中的API用法
        // 该行代码,实际调用的是[目标对象]的方法
        // 利用反射,调用[目标对象]的方法
        Object returnValue = method.invoke(target, args);
        return returnValue;
    }
}

CGLib动态代理
是通过子类继承父类的方式去实现的动态代理,不需要接口。

public class CgLibProxyFactory implements MethodInterceptor {
    /*** @param clazz * @return */
    public Object getProxyByCgLib(Class clazz) {
        // 创建增强器
        Enhancer enhancer = new Enhancer();
        // 设置需要增强的类的类对象
        enhancer.setSuperclass(clazz);
        // 设置回调函数
        enhancer.setCallback(this);
        // 获取增强之后的代理对象
        return enhancer.create();
    }

    /**
     * Object proxy:这是代理对象,也就是[目标对象]的子类
     * Method method:[目标对象]的方法
     * Object[] arg:参数
     * MethodProxy methodProxy:代理对象的方法
     */
    @Override
    public Object intercept(Object proxy, Method method, Object[] arg, MethodProxy methodProxy) throws Throwable {

        // 因为代理对象是目标对象的子类
        // 该行代码,实际调用的是父类目标对象的方法
        System.out.println("这是cglib的代理方法");
        // 通过调用子类[代理类]的invokeSuper方法,去实际调用[目标对象]的方法
        Object returnValue = methodProxy.invokeSuper(proxy, arg);
        // 代理对象调用代理对象的invokeSuper方法,而invokeSuper方法会去调用目标类的 invoke方法完成目标对象的调用
        return returnValue;
    }
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值