IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> 在JAVA中实现混型的两种方式 --《JAVA编程思想》 62 -> 正文阅读

[Java知识库]在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 任意类型进行方法调用。

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

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

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

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2021-10-24 14:48:21  更:2021-10-24 14:50:13 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 23:49:06-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码