一般递归自类型值的 Java 字段类型?

标签 java generics visitor-pattern self-type

给定一个类层次结构,其中基类定义了一个递归的自身类型:

abstract class A<T extends A<T>> { }

我如何声明另一个类(它在 T 中不应该是泛型的,因为这样的 T 在对象的生命周期内可能会发生变化),其字段可以包含 A 的任何子类?

以下不起作用:

public class B {
    //fails to compile, because the capture of ? is not sufficiently narrow
    private A<?> a;

    public <T extends A<T>> setA(T a) {
        this.a = a;
    }
}

-- 问题结束--

我注意到许多 StackOverflow 成员倾向于用“您首先为什么要这样做?”来解决某些难题。以下是我使用此模式的理由 - 您可以注意到 Java 标准库在其 Enum 类的定义中也使用递归自类型:Enum<E extends Enum<E>> .这个问题可以类似地问为“如何定义 Enum<?> 类型的字段。

理由示例:

abstract class A<T extends A<T>> {
    public abtract T self();
    public B<T> bify(Bifyer bifyer) {
        return bifyer.bify(self());
    }
}

带有子类:

class ASub1 extends A<ASub1> { 
    public ASub1 self() { return this; }
}       

class ASub2 extends A<ASub2> { 
    public ASub2 self() { return this; }
}       

绑定(bind)到并行类层次结构:

abstract class B<T extends A<T>> {
}

class BSub1<T extends A<T>> implements B<T> { }
class BSub2<T extends A<T>> implements B<T> { }
//and others

并且生成由 Bifyer 接口(interface)的实现管理的 B 实例:

interface Bifyer {
    B<ASub1> bify(ASub1 asub1);
    B<ASub2> bify(ASub2 asub2);        
}

此接口(interface)的实现可能会为 B 返回 BSub1 或 BSub2。这本质上是访问者模式的应用,其中 Bifyer 是访问者,但与标准访问者不同,accept 方法返回一个值而不是 void。这提供了一个模块化框架,其中可以指定不同的 Bifyer 实现来为 Bify 方法提供替代行为和返回类型 - 比如 B 的每个子类。

最佳答案

如果你用 A 绑定(bind)下面的通配符 ?,它应该可以工作:

public class B {
    private A<? extends A> a;

    public <T extends A<T>> void setA(T a) {
        this.a = a;
    }
}

关于一般递归自类型值的 Java 字段类型?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/695298/

相关文章:

java - 为什么每个实体都应该有单独的 MyBatis 映射器?

c# - 打印数组和任意类型列表的常用方法

C# 将 List<ObjBase> 转换为 List<Obj>

java - 如何声明具有多个具有重要关系的泛型类型的成员?

java - 在运行时动态选择方法;访问者模式或反射的替代方案

polymorphism - 为什么 std::visit 不能消除模板重载的歧义

java - Spring Jpa 更新 : Can not issue data manipulation statements with executeQuery()

java - 定制 epos 软件的支付系统集成

java.lang.ClassNotFoundException : org. hamcrest.Matchers 添加依赖项后 pom.xml

c++ - 我可以将一个类与模板中的另一个类关联起来(使用 C++17 变体)吗?