java的两种代理模式

本文深入探讨了代理模式的设计理念,通过具体实例展示了静态代理与动态代理(JDK代理)的实现方式,及其在不同场景下的应用优势与局限。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

代理模式是什么

代理模式是一种设计模式,简单说即是在不改变源码的情况下,实现对目标对象功能扩展

比如有个歌星对象叫RealStar,他十分会唱歌,但是这个歌手十分忙,他只想唱歌不愿意做其他的事情,因此你需要帮助他完成任务!

Star接口

package com.Geeksun.proxy.dynamicProxy;

public interface Star {
    void sign();
    
    void payMoney();
    
    void sing();
}

RealStar

package com.Geeksun.proxy.dynamicProxy;

public class RealStar implements Star {
    @Override
    public void sign() {
        System.out.println("我不会!");
    }

    @Override
    public void payMoney() {
        System.out.println("我不会!");
    }

    @Override
    public void sing() {
        System.out.println("sing");
    }
}

1.静态代理

这时候有一个代理对象

package com.Geeksun.proxy.staticProxy;

public class ProxyStar implements Star{

    private Star star;

    public ProxyStar(Star star){
        this.star = star;
    }

    @Override
    public void sign() {
        System.out.println("sign");
    }

    @Override
    public void payMoney() {
        System.out.println("payMoney");
    }

    @Override
    public void sing() {
        star.sing();
    }
}

 

 总结:其实这里做的事情无非就是,创建一个代理类ProxyStar,继承了Star接口并实现了其中的方法。只不过这种实现特意包含了目标对象的方法,正是这种特征使得看起来像是“扩展”了目标对象的方法。假使代理对象中只是简单地对sing方法做了另一种实现而没有包含目标对象的方法,也就不能算作代理模式了。所以这里的包含是关键。

  缺点:这种实现方式很直观也很简单,但其缺点是代理对象必须提前写出,如果接口层发生了变化,代理对象的代码也要进行维护。如果能在运行时动态地写出代理对象,不但减少了一大批代理类的代码,也少了不断维护的烦恼,不过运行时的效率必定受到影响。这种方式就是接下来的动态代理。

2.动态代理(也叫JDK代理)

 跟静态代理的前提一样,依然是对Star对象进行扩展

StarHandler

package com.Geeksun.proxy.dynamicProxy;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;

public class StarHandler implements InvocationHandler {
    private Star star;

    public StarHandler(Star star){
        this.star = star;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        Object object = null;
        System.out.println("订票");
        if(method.getName().equals("sing")){
            object = method.invoke(star,args);
        }
        System.out.println("收钱");
        return object;
    }
}

Client

package com.Geeksun.proxy.dynamicProxy;

import java.lang.reflect.Proxy;

public class Client {

    public static void main(String[] args) {
        StarHandler starHandler = new StarHandler(new RealStar());

        Star proxy = (Star) Proxy.newProxyInstance(ClassLoader.getSystemClassLoader(),new Class[]{Star.class},starHandler);
        proxy.sign();
        proxy.sing();
    }

}

调用Proxy类的静态方法newProxyInstance即可,该方法会返回代理类对象

static Object newProxyInstance(ClassLoader loader, Class<?>[] interfaces,InvocationHandler h )

接收的三个参数依次为:

  • ClassLoader loader:指定当前目标对象使用类加载器,写法固定
  • Class<?>[] interfaces:目标对象实现的接口的类型,写法固定
  • InvocationHandler h:事件处理接口,需传入一个实现类,一般直接使用匿名内部类
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值