《Effective Java》解读第38条:用接口模拟可扩展的枚举

第38条:用接口模拟可扩展的枚举

主要解决Java枚举的一个局限性:枚举类型不能被继承。但在某些场景下,我们希望枚举具有一定的扩展性,允许第三方添加新的枚举常量。

解决方法,通过接口来拓展枚举。

接口 + 枚举

虽然枚举类型不是可扩展的,但接口类型却是可扩展的,它是用来表示 API 中的操作的接口类型,可以定义另 个枚举类型,它实现这个接口,并用这个新类型的实例代替基本类型。

java 复制代码
// 步骤1:定义操作接口
public interface Operation {
    double apply(double x, double y);
}

// 步骤2:基础枚举实现接口
public enum BasicOperation implements Operation {
    PLUS("+") {
        public double apply(double x, double y) { return x + y; }
    },
    MINUS("-") {
        public double apply(double x, double y) { return x - y; }
    };
    
    private final String symbol;
    BasicOperation(String symbol) { this.symbol = symbol; }
}

// 步骤3:扩展枚举(新的枚举实现同一接口)
public enum ExtendedOperation implements Operation {
    TIMES("*") {
        public double apply(double x, double y) { return x * y; }
    },
    DIVIDE("/") {
        public double apply(double x, double y) { return x / y; }
    };
    
    private final String symbol;
    ExtendedOperation(String symbol) { this.symbol = symbol; }
}

使用

java 复制代码
public class Main {
    public static void main(String[] args) {
        System.out.println(BasicOperation.PLUS.apply(1, 2));
        System.out.println(BasicOperation.MINUS.apply(1, 2));
        System.out.println(ExtendedOperation.TIMES.apply(1, 2));
        System.out.println(ExtendedOperation.DIVIDE.apply(1, 2));

        test(BasicOperation.class, 1, 2);
    }

    // 方法可以接受任何实现了Operation接口的枚举
    public static <T extends Enum<T> & Operation> void test(
            Class<T> opEnumType, double x, double y) {
        for (Operation op : opEnumType.getEnumConstants()) {
            System.out.printf("%f %s %f = %f%n",
                    x, op, y, op.apply(x, y));
        }
    }
}

也可以通过集合参数使用

java 复制代码
// 可以混合使用不同枚举类型的操作
public static void testAll(Collection<? extends Operation> operations,
        double x, double y) {
    for (Operation op : operations) {
        System.out.printf("%f %s %f = %f%n",
            x, op, y, op.apply(x, y));
    }
}

// 混合使用
testAll(Arrays.asList(BasicOperation.PLUS, 
                     ExtendedOperation.TIMES), 10, 5);

总结

虽然无法编写可扩展的枚举类型,却可以通过编写接口以及实现该接口的基础枚举类型来对它进行模拟.

相关推荐
MY_TEUCK11 小时前
Sealos 平台部署实战指南:结合 Cursor 与版本发布流程
java·人工智能·学习·aigc
2401_8734794011 小时前
如何利用IP查询定位识别电商刷单?4个关键指标+工具配置方案
开发语言·tcp/ip·php
我爱cope11 小时前
【从0开始学设计模式-10| 装饰模式】
java·开发语言·设计模式
菜鸟学Python12 小时前
Python生态在悄悄改变:FastAPI全面反超,Django和Flask还行吗?
开发语言·python·django·flask·fastapi
朝新_12 小时前
【Spring AI 】图像与语音模型实战
java·人工智能·spring
RH23121112 小时前
2026.4.16Linux 管道
java·linux·服务器
zmsofts13 小时前
java面试必问13:MyBatis 一级缓存、二级缓存:从原理到脏数据,一篇讲透
java·面试·mybatis
浪浪小洋13 小时前
c++ qt课设定制
开发语言·c++
charlie11451419113 小时前
嵌入式C++工程实践第16篇:第四次重构 —— LED模板,从通用GPIO到专用抽象
c语言·开发语言·c++·驱动开发·嵌入式硬件·重构
故事和你9113 小时前
洛谷-数据结构1-4-图的基本应用1
开发语言·数据结构·算法·深度优先·动态规划·图论