在我的spring应用程序上下文文件中,我有如下内容:

<util:map id="someMap" map-class="java.util.HashMap" key-type="java.lang.String" value-type="java.lang.String">
    <entry key="some_key" value="some value" />
    <entry key="some_key_2" value="some value" />   
</util:map>

在java类中,实现看起来像:

private Map<String, String> someMap = new HashMap<String, String>();
someMap = (HashMap<String, String>)getApplicationContext().getBean("someMap");

在Eclipse中,我看到一个警告说:

类型安全:未检查从Object转换到HashMap<String,String>

出了什么问题?


当前回答

避免未检查警告的解决方案:

class MyMap extends HashMap<String, String> {};
someMap = (MyMap)getApplicationContext().getBean("someMap");

其他回答

另一种解决方案是,如果您发现自己经常强制转换同一个对象,而又不想让@ suppresswarnings(“unchecked”)乱写代码,那么可以使用注释创建一个方法。通过这种方式,您可以集中转换,并希望减少出错的可能性。

@SuppressWarnings("unchecked")
public static List<String> getFooStrings(Map<String, List<String>> ctx) {
    return (List<String>) ctx.get("foos");
}

如上面的消息所示,List不能区分List<Object>和List<String>或List<Integer>。

我已经解决了类似问题的错误信息:

List<String> strList = (List<String>) someFunction();
String s = strList.get(0);

与以下:

List<?> strList = (List<?>) someFunction();
String s = (String) strList.get(0);

解释:第一个类型转换验证对象是一个List,而不关心其中持有的类型(因为我们不能在List级别验证内部类型)。现在需要进行第二次转换,因为编译器只知道List包含某种类型的对象。这将在访问List时验证每个对象的类型。

问题是强制转换是一个运行时检查-但由于类型擦除,在运行时,对于任何其他Foo和Bar, HashMap<String,String>和HashMap<Foo,Bar>之间实际上没有区别。

使用@SuppressWarnings(“未选中”),捏住鼻子。哦,还有在Java中为具体化泛型发起的运动:)

您得到这条消息是因为getBean返回了一个Object引用,并且您正在将它强制转换为正确的类型。Java 1.5给出了一个警告。这就是使用Java 1.5或更高版本代码的本质。Spring有类型安全版本

someMap=getApplicationContext().getBean<HashMap<String, String>>("someMap");

在待办事项清单上。

如果您真的想摆脱警告,可以做的一件事是创建一个从泛型类扩展而来的类。

比如,如果你想用

private Map<String, String> someMap = new HashMap<String, String>();

您可以像这样创建一个新类

public class StringMap extends HashMap<String, String>()
{
    // Override constructors
}

那么当你使用

someMap = (StringMap) getApplicationContext().getBean("someMap");

编译器确实知道(不再是泛型)类型是什么,并且不会有警告。这可能并不总是完美的解决方案,有些人可能会认为这违背了泛型类的目的,但你仍然重用了泛型类中的所有相同代码,你只是在编译时声明了你想使用的类型。