Java中的枚举和自限定类型的泛型

Java中所有的enum都继承自Java.lang.Enum类,而Enum类的定义如下:

public abstract class Enum<E extends Enum<E>> implements Comparable<E>, Serializable {
    // ...
}

从定义可以看到,Enum类使用了自限定类型的泛型。

自限定的类型参数有和意义呢?它可以保证泛型参数必须与正在被定义的类相同(或有相同的基类),例如:

public class Test {
    public static void main(String[] args) {
        MyEnum1 enum1 = new MyEnum1();
        MyEnum1 enum11 = new MyEnum1();

        MyEnum2 enum2 = new MyEnum2();
        MyEnum2 enum22 = new MyEnum2();

        MyEnum3 enum3 = new MyEnum3();
        MyEnum3 enum33 = new MyEnum3();

        enum1.compareTo(enum11);
        enum2.compareTo(enum22);
        enum3.compareTo(enum2);

    }
}

abstract class AbsEnum<E extends AbsEnum<E>> implements Comparable<E> {}


class MyEnum1 extends AbsEnum<MyEnum1> {
    @Override
    public int compareTo(MyEnum1 o) {
        return 0;
    }
}

class MyEnum2 extends AbsEnum<MyEnum2> {
    @Override
    public int compareTo(MyEnum2 o) {
        return 0;
    }
}

class MyEnum3 extends AbsEnum<MyEnum2> {
    @Override
    public int compareTo(MyEnum2 o) {
        return 0;
    }
}

AbsEnum是一个使用自限定类型的基类,AbsEnum要求所有继承自AbsEnum的子类的泛型参数必须时AbsEnum的子类。这样就相当于给泛型提供了一个边界。

所以,当我们定义了一个泛型类的时候

public enum Month {
    JAN, FEB, MAR;
}

编译器做的事就是把这个enum编译成Enum类的子类,使用反编译工具可得到:

final class Month extends java.lang.Enum<Month> {
    public static final Month JAN;
    public static final Month FEB;
    public static final Month Mar;
}

如此就保证了Month这个枚举类的实例只能和Month的实例进行比较(如上例的MyEnum1和MyEnum2),而不会出现两个不同的枚举类进行比较(如上例的MyEnum3)

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Java,获取lambda表达式的泛型类型是不直接支持的。Lambda表达式在编译时被转换为函数式接口的实例,而函数式接口可能会定义一个或多个泛型参数。 在Java,我们不能直接获取lambda表达式的泛型类型,因为泛型类型被擦除了(type erasure)。在运行时,泛型类型信息是不可用的。Lambda表达式只能通过函数式接口定义的方法来访问参数和返回类型。 然而,可以通过一些间接的方式来获取lambda表达式的泛型类型。例如,可以使用反射来获取函数式接口的类型信息,并进一步获取泛型参数的类型。以下是一个示例: ```java import java.lang.reflect.ParameterizedType; import java.lang.reflect.Type; public class Main { public static void main(String[] args) { MyInterface<String> myObj = s -> System.out.println(s); Type type = myObj.getClass().getGenericInterfaces()[0]; if (type instanceof ParameterizedType) { ParameterizedType parameterizedType = (ParameterizedType) type; Type[] typeArguments = parameterizedType.getActualTypeArguments(); for (Type typeArgument : typeArguments) { System.out.println(typeArgument); } } } } interface MyInterface<T> { void doSomething(T t); } ``` 在上述示例,我们创建了一个使用String类型的lambda表达式,并使用反射来获取该lambda表达式实现的函数式接口的泛型类型。需要注意的是,这种方式只适用于已知函数式接口的情况,并且只能获取泛型参数的类型,无法获取具体的类型参数。 总结起来,虽然在Java不能直接获取lambda表达式的泛型类型,但可以通过反射间接获取函数式接口的泛型信息。然而,这种方式的适用范围有限,并且可能会增加代码的复杂性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值