正确使用“<t extends =”“superclass =”“>”</t>

时间:2011-02-17 14:58:29

标签: java generics

我不熟悉“泛型”。它是否正确使用“<T extends SuperClass>”?您是否同意使用泛型后的代码更好?

Before using Generics
=================================================

public abstract class SuperSample {

    public void getSomething(boolean isProcessA) {
        doProcessX();
        if(isProcessA){
            doProcessY(new SubASample());
        }else{
            doProcessY(new SubBSample());
        }
    }

    protected abstract void doProcessX();

    protected void doProcessY(SubASample subASample) {
        // Nothing to do
    }

    protected void doProcessY(SubBSample subBSample) {
        // Nothing to do
    }

}

public class SubASample extends SuperSample {

    @Override
    protected void doProcessX() {
        System.out.println("doProcessX in SubASample");
    }

    @Override
    protected void doProcessY(SubASample subASample) {
        System.out.println("doProcessY in SubASample");
    }

}

public class Sample {

    public static void main(String[] args) {
        SubASample subASample = new SubASample();
        subASample.getSomething(true);
    }

}

After using Generics
=================================================

public abstract class SuperSample {

    public void getSomething(boolean isProcessA) {
        doProcessX();
        if(isProcessA){
            doProcessY(new SubASample());
        }else{
            doProcessY(new SubBSample());
        }
    }

    protected abstract void doProcessX();

    protected abstract <T extends SuperSample> void doProcessY(T subSample);

}

public class SubASample extends SuperSample {

    @Override
    protected void doProcessX() {
        System.out.println("doProcessX in SubASample");
    }

    @Override
    protected <T extends SuperSample> void doProcessY(T subSample) {
        System.out.println("doProcessY in SubASample");
    }

}

public class Sample {

    public static void main(String[] args) {
        SubASample subASample = new SubASample();
        subASample.getSomething(true);
    }

}

2 个答案:

答案 0 :(得分:4)

如果你想做我想做的事情,我认为这不是正确的方法(*)。如果您希望每个子类都需要实现一个处理它自己类型的方法,那么您可以使用CRTP技巧:

abstract class Super<S extends Super<S>> {
   abstract void process(S s);
}

class SubA extends Super<SubA> {
   void process(SubA s){ /* do something */ }
}

class SubB extends Super<SubB> {
   void process(SubB s){ /* do something */ }
}

请注意,此模式强制执行子类的通用签名,例如class SubA extends Super<SubB>无法编译。

顺便说一下,Java本身在java.lang.Enum中使用了这个技巧。

(*)如果这不是您要强制执行的行为,请澄清。

答案 1 :(得分:1)

使用是正确的。这意味着您将类型T限制为SuperSample的子类。对于第二个答案,是的,我认为使用generecis的代码更好,因为它可以防止错误的类转换,例如容器(List ...)。但事实上,Java中的泛型只是语法,因此在运行时会被删除。