在JAVA中实现混型的两种方式 --《JAVA编程思想》 62

混型的基本概念是混合多个类的能力,以产生一个可以表示混型中所有类型的类。

我们来看个简单的需求:现在需要把获取当前时间戳、获取计数器和存取某个值的三个类合成一个混型。

1. 通过接口实现

首先,我们可以直接通过接口的形式来满足需求。

public interface TimeStamped {
    long getStamp();
}
public class TimeStampedImp implements TimeStamped {

    @Override
    public long getStamp() {
        return new Date().getTime();
    }

}
public interface SerialNumbered {
    long getSerialNumber();
}
public class SerialNumberedImp implements SerialNumbered {

    private static Long counter = 1L;

    private final long serialNumber = counter++;

    @Override
    public long getSerialNumber() {
        return serialNumber;
    }
    
}
public interface Basic {

    void set(String val);

    String get();

}

public class BasicImp implements Basic {

    private String val;

    @Override
    public void set(String val) {
        this.val = val;
    }

    @Override
    public String get() {
        return val;
    }

}
public class Mixin extends BasicImp implements TimeStamped, SerialNumbered {

    private TimeStamped timeStamped = new TimeStampedImp();

    private SerialNumbered serialNumbered = new SerialNumberedImp();

    @Override
    public long getSerialNumber() {
        return serialNumbered.getSerialNumber();
    }

    @Override
    public long getStamp() {
        return timeStamped.getStamp();
    }
    
}
public class Mixins {

    public static void main(String[] args) {
        Mixin mixin = new Mixin();
        mixin.set("Today is a good day!");
        System.out.println(mixin.get());
        System.out.println(mixin.getStamp());
        System.out.println(mixin.getSerialNumber());
    }
    
}
Today is a good day!
1634979111046
1

TimeStamped 和 TimeStampedImp 为获取当前时间戳的接口和实现类;

SerialNumbered 和 SerialNumberedImp 为获取计数器的接口和实现类;

Basic 和 BasicImp 为设置和获取 String 类型值得接口和实现类。

Mixin 继承 BasicImp 获得存取值得能力,再通过实现 TimeStamped 和 SerialNumbered 两个接口,在内部注入两个对应接口的实现类,从而使得 Mixin 同时混合了三个类的能力。

使用此种方式实现混型都要求每种被混入的类型在 Mixin 中有对应的域,还得在混型类中编写方法将调用转发给具体的实现类,业务越复杂代码量就会越大

2. 通过动态代理来实现混合

其次,我们可以通过动态代理来实现。由于动态代理的限制,每个被混入的类都要求为接口。

public class TwoTuple<A, B> {


    public TwoTuple(A first, B second) {
        this.first = first;
        this.second = second;
    }

    private A first;

    private B second;

    public A getFirst() {
        return first;
    }

    public B getSecond() {
        return second;
    }
    
}
public class MixinProxy implements InvocationHandler {

    Map<String, Object> delegatesByMethod;

    public MixinProxy(TwoTuple<Object, Class<?>>... pairs) {
        delegatesByMethod = new HashMap<>();
        //将需要代理的方法放入Map
        for (TwoTuple<Object, Class<?>> pair : pairs) {
            Method[] methods = pair.getSecond().getMethods();
            for (Method method : methods) {
                String methodName = method.getName();
                if (!delegatesByMethod.containsKey(methodName)) {
                    delegatesByMethod.put(methodName, pair.getFirst());
                }
            }
        }
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        String methodName = method.getName();
        //返回实际处理业务的对象
        Object delegate = delegatesByMethod.get(methodName);
        return method.invoke(delegate, args);
    }

    public static Object newInstance(TwoTuple... pairs) {
        //收集代理的接口
        Class[] interfaces = new Class[pairs.length];
        for (int i = 0; i < interfaces.length; i++) {
            interfaces[i]= (Class) pairs[i].getSecond();
        }
        ClassLoader cl=pairs[0].getFirst().getClass().getClassLoader();
        return Proxy.newProxyInstance(cl,interfaces,new MixinProxy(pairs));
    }

}
public class DynamicProxyMixin {

    public static void main(String[] args) {
        Object mixin = MixinProxy.newInstance(new TwoTuple(new BasicImp(), Basic.class),
                new TwoTuple(new TimeStampedImp(),
                TimeStamped.class), new TwoTuple(new SerialNumberedImp(), SerialNumbered.class));
        Basic b= (Basic) mixin;
        TimeStamped t = (TimeStamped) mixin;
        SerialNumbered s = (SerialNumbered) mixin;
        b.set("dynamicProxy");
        System.out.println(b.get());
        System.out.println(t.getStamp());
        System.out.println(s.getSerialNumber());
    }

}
dynamicProxy
1634985088839
1

TwoTuple 存储两个类型参数,第一个为被代理接口的实现类,第二个为被代理接口。

在 MixinProxy 代理类的构造方法中,将需要代理的方法作为 key ,实现代理方法的实现类作为 value 存入 delegatesByMethod 中。

通过 newInstance() 方法获取实例时,传入第一个接口实现类的类加载器(任意一个接口实现类的加载器都行,传入第一个是防止因下标越界引起异常)、需要被代理的接口、实现 InvocationHandler 接口的类,从而返回代理对象。因代理对象同时代理了多组接口,故可以转换为 Basic 、TimeStamped 、SerialNumbered 任意类型进行方法调用。

倘若我们需要添加新的类型至混型之中,只需要拓展对应的接口。相比直接通过接口实现混型,此方式的代码量更少,也更为灵活。

本次分享至此结束,希望本文对你有所帮助,若能点亮下方的点赞按钮,在下感激不尽,谢谢您的【精神支持】。

若有任何疑问,也欢迎与我交流,若存在不足之处,也欢迎各位指正!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

BaymaxCS

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值