Java:使用泛型和地图而不使用Casting / @SuppressWarnings

时间:2016-06-24 16:59:56

标签: java generics casting pecs

我现在多次遇到过这个问题,总是通过一些强制转换和@SuppressWarnings注释来解决这个问题。

相关接口/抽象类:

public abstract class Data { }

public interface DataOperations {
    boolean isValid();
}

public interface DataOperationsFactory<T extends Data> {
    Class<T> getDataClass();
    DataOperations getOperations(T data);
}

示例实施:

public class DataImpl1 extends Data {
    public String foo;
}

public class DataImpl1Operations implements DataOperations {
    private DataImpl1 data;
    public DataImpl1Operations(DataImpl1 data) {
        this.data = data;
    }
    public boolean isValid() {
        return data.foo != null;
    }
}

public class DataImpl1OperationsFactory extends DataOperationsFactory<DataImpl1> {
    public Class<DataImpl1> getDataClass() {
        return DataImpl1.class;
    }
    DataOperations getOperations(DataImpl1 data) {
        return new DataImpl1Operations(data);
    }
}

使用此模式,我可以决定是否每次都需要创建new DataImpl1Operations。或者也许使用final static NO_OP实现或者你有什么。

代码:

现在我想将所有这些工厂放在Map<Class<T>, DataOperationsFactory<T>>(构造函数)中。然后从中读取(getOps方法)。

public class Test {
    Map<Class<?>, DataOperationsFactory<?>> map;

    public Test(List<DataOperationsFactory<?>> fs) {
        for(DataOperationsFactory<?> f : fs) {
            map.put(f.getDataClass(), f);
        }
    }

    @SuppressWarnings("unchecked")
    public <T extends Data> DataOperations getOps(T data) {
        // --> Here I need to do an unchecked cast <--
        DataOperationsFactory<? super T> f =
                (DataOperationsFactory<? super T>) map.get(data.getClass());
        return f.getOperations(data);
    }
}

如果不进行未选中的投射,有没有办法做到这一点?

1 个答案:

答案 0 :(得分:3)

您可以委托捕获类型的私有方法,因此可以用它可靠地转换为正确的Data子类:

Map<Class<?>, DataOperationsFactory<?>> map;

// Unchanged
public Test(List<DataOperationsFactory<?>> fs) {
    for(DataOperationsFactory<?> f : fs) {
        map.put(f.getDataClass(), f);
    }
}

public DataOperations getOps(Data data) {
    DataOperationsFactory<?> f = map.get(data.getClass());
    return getOperations(f, data);
}

private static <T extends Data> DataOperations getOperations(DataOperationsFactory<T> f,
                                                             Data data) {
    return f.getOperations(f.getDataClass().cast(data));
}