当前位置: 主页 > JAVA语言

代理模式java-JavaRealSubject.javaProxy.–虚拟代理测试

发布时间:2023-06-17 14:49   浏览次数:次   作者:佚名

一、概念二、代理模式的几种变体1、远程代理

之前在看《Spring 实战》的时候,接触到了RPC机制。现在在读设计模式的时候代理模式java,才发现 RPC 在设计模式中还有另外一个名字 —— 远程代理。所以这里就不展开讲了,详细了解可参考之前的文章。

2、虚拟代理

虚拟代理作为创建开销大的对象的代表。虚拟代理经常直到我们真正需要一个对象的时候才创建它。当对象在创建前和创建中的时,由虚拟代理来扮演对象的替身。对象创建后,代理就会将请求直接委托给对象。

我们来看一个简单Demo,对于虚拟代理和普通代理的区别就一目了然了~~

Subject.java

java代理模式_代理模式java_代理模式 动态代理

/**
 * @Description: 主题接口
 */
public interface Subject {
    void request();
}

RealSubject.java

/**
 * @Description:具体主题 — 真正做事的地方
 */
public class RealSubject implements Subject {
    @Override
    public void request() {
        System.out.println("this is real subject.");
    }
}

Proxy.java – 普通代理

/**
 * @Description: 普通代理
 */
public class Proxy implements Subject {
    private Subject subject;
    public Proxy(Subject subject) {
        this.subject = subject;
    }
    @Override
    public void request() {
        subject.request();
    }
}

代理模式 动态代理_java代理模式_代理模式java

VirtualProxy.java – 虚拟代理

/**
 * @Description: 虚拟代理
 */
public class VirtualProxy implements Subject {
    private Subject subject;
    @Override
    public void request() {
        // 在真正使用的时候才创建对象
        if (subject == null) {
            subject = new RealSubject();
        }
        subject.request();
    }
}

测试

public class Test {
    public static void main(String[] args) {
        //1、普通代理
        Subject subject = new RealSubject();
        Proxy proxy = new Proxy(subject);
        proxy.request();
        //2、虚拟代理
        Subject virtualProxy = new VirtualProxy();
        virtualProxy.request();
    }
}

3、保护代理

代理模式 动态代理_java代理模式_代理模式java

Java 在java.lang.reflect包中有自己的代理支持,利用这个包你可以在运行时动态的创建一个代理类,并将方法的调用转发到你说指定的类。因为实际的代理类是在运行时创建的,我们称这个Java技术为:动态代理。

我们要利用 Java 的动态代理创建一个保护代理实现。所谓保护代理,有点像权限控制系统,代理会拦截所有的请求并判断该次请求是否有权限访问对应的资源。

代理模式java_代理模式 动态代理_java代理模式

现在我们用保护代理简单的做一个权限控制,具体表现为:顾客可以查看商品的名字、详情、价格,而卖家可以看到商品的所有信息,包括利润等。

商品 — Goods.java

代理模式 动态代理_代理模式java_java代理模式

public interface Goods {
    
    /**
     * @Description 商品的名字、详情、价格 —— 顾客可以查看
     */
    String getName(Long id);
    String getDetail(Long id);
    BigDecimal getPrice(Long id);
    /**
     * @Description 商品的利润 —— 只有卖家可以看到
     */
    BigDecimal getProfit(Long id);
}

InvocationHandler

InvocationHandler 的工作是响应代理的任何调用。你可以把 InvocationHandler 想成是代理收到方法调用后代理模式java,请求做实际工作的对象。就是在 InvocationHandler 中,我们对请求进行筛选,判断是否有权限访问相应的资源。

/**
 * @Description: 顾客InvocationHandler — 看不到商品利润信息
 */
public class CustomerInvocationHandler implements InvocationHandler {
    private Goods goods;
    public CustomerInvocationHandler(Goods goods) {
        this.goods = goods;
    }
    
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        if ("getName".equals(method.getName())
                || "getDetail".equals(method.getName())
                || "getPrice".equals(method.getName())) {
            return method.invoke(goods, args);
        }
        if ("getProfit".equals(method.getName())) {
            throw new IllegalAccessException();
        }
        return null;
    }
}

/**
 * @Description: 商家InvocationHandler — 可以看到所有信息
 */
public class SellerInvocationHandler implements InvocationHandler {
    private Goods goods;
    public SellerInvocationHandler(Goods goods) {
        this.goods = goods;
    }
    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        return method.invoke(goods, args);
    }
}

代理模式java_代理模式 动态代理_java代理模式

测试

public class Test {
    
    public static void main(String[] args) {
        Goods goods = new Clothes();
        // 创建顾客代理
        Goods customer = (Goods) Proxy.newProxyInstance(goods.getClass().getClassLoader(),
                goods.getClass().getInterfaces(),
                new CustomerInvocationHandler(goods));
        // 创建卖家代理
        Goods seller = (Goods) Proxy.newProxyInstance(goods.getClass().getClassLoader(),
                goods.getClass().getInterfaces(),
                new SellerInvocationHandler(goods));
        // 判断某个类是不是代理类
        System.out.println(Proxy.isProxyClass(customer.getClass()));
        
        // 卖家代理调用
        System.out.println("-------卖家代理调用开始--------");
        System.out.println(seller.getName(800001L));
        System.out.println(seller.getDetail(800001L));
        System.out.println(seller.getPrice(800001L).doubleValue());
        System.out.println(seller.getProfit(800001L).doubleValue());
        System.out.println("-------卖家代理调用结束--------");
        // 顾客代理调用
        System.out.println("-------顾客代理调用开始--------");
        System.out.println(customer.getName(800001L));
        System.out.println(customer.getDetail(800001L));
        System.out.println(customer.getPrice(800001L).doubleValue());
        System.out.println(customer.getProfit(800001L).doubleValue());
        System.out.println("-------顾客代理调用结束--------");
    }
}

三、总结

演示源代码:

tips:transient 关键字,告诉 jvm 不要序列化这个字段。

四、寄语

年前的最后一篇文章了,祝愿大家新年快乐。愿大家早日实现财务自由!愿大家此生想得到的都拥有,得不到的都释怀!