java - 具有通用返回值的方法

标签 java generics methods parameters

我在编写具有通用返回值的方法时遇到困难。我希望有一种方法,当使用 MachineMorphology 实例调用时,返回一个 M 类型的实例,其中 MMachineMorphology 泛型类型。

这可以实现吗?

机器形态学.java

public interface MachineMorphology<M extends Machine> {

    Class<? extends M> getClazz();
}

ExtrusionMachinesMorphology.java

public final class ExtrusionMachinesMorphology<M extends Machine> implements MachineMorphology<M> {
    public static final ExtrusionMachinesMorphology<Doser> DOSER = new ExtrusionMachinesMorphology<>(Doser.class);
    public static final ExtrusionMachinesMorphology<Extruder> EXTRUDER = new ExtrusionMachinesMorphology<>(Extruder.class);
    public static final ExtrusionMachinesMorphology<Blower> BLOWER = new ExtrusionMachinesMorphology<>(Blower.class);
    public static final ExtrusionMachinesMorphology<Die> DIE = new ExtrusionMachinesMorphology<>(Die.class);
    public static final ExtrusionMachinesMorphology<HaulOff> HAULOFF = new ExtrusionMachinesMorphology<>(HaulOff.class);
    public static final ExtrusionMachinesMorphology<Winder> WINDER = new ExtrusionMachinesMorphology<>(Winder.class);
    public static final ExtrusionMachinesMorphology<FilmMeasurer> MEASUSER = new ExtrusionMachinesMorphology<>(FilmMeasurer.class);
    public static final ExtrusionMachinesMorphology<AirRing> AIR_RING = new ExtrusionMachinesMorphology<>(AirRing.class);
    public static final ExtrusionMachinesMorphology GEC = null;
    public static final ExtrusionMachinesMorphology UNDEFINED = null;

    private final Class<M> clazz;

    public ExtrusionMachinesMorphology(Class<M> clazz) {
        this.clazz = clazz;
    }

    public Class<? extends M> getClazz() {
        return clazz;
    }
}

AbstractExtrusionPlant.java

public abstract class AbstractPlant<T extends PlantType, S extends PlantStatus, O extends PlantOperatorStatus, A extends Alarm> 
    extends ListenableObject<AlarmsListener<Machine, A>> {

    [...]

    private volatile HashMap<MachineMorphology, List<? extends Machine>> machines = new HashMap<>();

    public synchronized <M extends Machine> M getMachine(MachineMorphology<M> morphology, int unitId) {
        return getMachineByUnitId(getMachinesByMorphology(morphology), unitId);
    }

    public synchronized <M extends Machine> M getMachineByUnitId(List<M> machines, int unitId) {
        return machines.stream().filter(machine -> machine.getUnitId() == unitId).findFirst().orElse(null);
    }

    public synchronized <M extends Machine> List<M> getMachinesByMorphology(MachineMorphology<M> morphology) {
        return (List<M>) machines.get(morphology);
    }

    [...]
}

这应该是使用方法

Doser doser = plant.getMachine(ExtrusionMachinesMorphology.DOSER, 1);

但是编译器想要:

 Doser doser = (Doser)plant.getMachine(ExtrusionMachinesMorphology.DOSER, 1);

最佳答案

我相信您是在问 Map 是否可以返回具有不同泛型类型的对象。答案是否定的,这是不可能的,但是你可以在返回之前使用Morphology的类型来转换List:

public synchronized <M extends Machine> List<M> getMachinesByMorphology(MachineMorphology<M> morphology) {
    Class<? extends M> machineType = morphology.getClazz();
    return machines.get(morphology).stream().map(machineType::cast).collect(Collectors.toList());
}

更新:这是我用于测试可编译性的完整文件:

import java.util.HashMap;
import java.util.List;
import java.util.stream.Collectors;

class Machine {
    int getUnitId() { return 0; }
}

class PlantType { }
class PlantStatus { }
class PlantOperatorStatus { }
class Alarm { }
class Doser extends Machine { }
class Extruder extends Machine { }
class Blower extends Machine { }
class Die extends Machine { }
class HaulOff extends Machine { }
class Winder extends Machine { }
class FilmMeasurer extends Machine { }
class AirRing extends Machine { }

interface AlarmsListener<M, A> { }
class ListenableObject<T> { }

interface MachineMorphology<M extends Machine> {

    Class<? extends M> getClazz();
}

final class ExtrusionMachinesMorphology<M extends Machine> implements MachineMorphology<M> {
    public static final ExtrusionMachinesMorphology<Doser> DOSER = new ExtrusionMachinesMorphology<>(Doser.class);
    public static final ExtrusionMachinesMorphology<Extruder> EXTRUDER = new ExtrusionMachinesMorphology<>(Extruder.class);
    public static final ExtrusionMachinesMorphology<Blower> BLOWER = new ExtrusionMachinesMorphology<>(Blower.class);
    public static final ExtrusionMachinesMorphology<Die> DIE = new ExtrusionMachinesMorphology<>(Die.class);
    public static final ExtrusionMachinesMorphology<HaulOff> HAULOFF = new ExtrusionMachinesMorphology<>(HaulOff.class);
    public static final ExtrusionMachinesMorphology<Winder> WINDER = new ExtrusionMachinesMorphology<>(Winder.class);
    public static final ExtrusionMachinesMorphology<FilmMeasurer> MEASUSER = new ExtrusionMachinesMorphology<>(FilmMeasurer.class);
    public static final ExtrusionMachinesMorphology<AirRing> AIR_RING = new ExtrusionMachinesMorphology<>(AirRing.class);
    public static final ExtrusionMachinesMorphology<?> GEC = null;
    public static final ExtrusionMachinesMorphology<?> UNDEFINED = null;

    private final Class<M> clazz;

    public ExtrusionMachinesMorphology(Class<M> clazz) {
        this.clazz = clazz;
    }

    public Class<? extends M> getClazz() {
        return clazz;
    }
}

public abstract class AbstractPlant<T extends PlantType, S extends PlantStatus, O extends PlantOperatorStatus, A extends Alarm> 
    extends ListenableObject<AlarmsListener<Machine, A>> {

    private volatile HashMap<MachineMorphology<? extends Machine>, List<? extends Machine>> machines = new HashMap<>();

    public synchronized <M extends Machine> M getMachine(MachineMorphology<M> morphology,
                                                         int unitId) {
        return getMachineByUnitId(getMachinesByMorphology(morphology), unitId);
    }

    public synchronized <M extends Machine> M getMachineByUnitId(List<M> machines, int unitId) {
        return machines.stream().filter(machine -> machine.getUnitId() == unitId).findFirst().orElse(null);
    }

    public synchronized <M extends Machine> List<M> getMachinesByMorphology(MachineMorphology<M> morphology) {
        Class<? extends M> machineType = morphology.getClazz();
        return machines.get(morphology).stream().map(machineType::cast).collect(Collectors.toList());
    }

    public static void main(String[] args) {
        AbstractPlant<PlantType, PlantStatus, PlantOperatorStatus, Alarm> plant =
            new AbstractPlant<PlantType, PlantStatus, PlantOperatorStatus, Alarm>() { };

        Doser doser = plant.getMachine(ExtrusionMachinesMorphology.DOSER, 1);
    }
}

关于java - 具有通用返回值的方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39167117/

相关文章:

java - Cucumber Scenario Outline 只运行了一次,即使它有 3 个场景

java - 对于这样的事情,ANTLR 是正确的方法吗?

java - GWT 电子表格单元格

c# - 无法将泛型类型转换为 C# 中的列表

Swift 通用完成处理程序

iphone - 在特定时间调用方法

python - `classmethod` 和元类方法有什么区别?

java - 为什么这与 String.java 中的另一个对象检查相等?

java - Java 中泛型类型的行为

javascript - JavaScript 中的所有函数本质上都是方法吗?