是否可以在Java中创建泛型类型的实例?我正在考虑基于我所看到的答案是no(由于类型擦除),但如果有人能看到我缺少的东西,我会感兴趣:
class SomeContainer<E>
{
E createContents()
{
return what???
}
}
Run Code Online (Sandbox Code Playgroud)
编辑:事实证明,超级类型标记可用于解决我的问题,但它需要大量基于反射的代码,如下面的一些答案所示.
我会把这个开放一段时间,看看是否有人提出了与Ian Robertson的Artima文章截然不同的任何东西.
我想通过Google Gson传输列表对象,但我不知道如何反序列化泛型类型.
我看后试了一下这个(BalusC的答案):
MyClass mc = new Gson().fromJson(result, new List<MyClass>(){}.getClass());
Run Code Online (Sandbox Code Playgroud)
但是我在eclipse中得到一个错误,说"类型新List(){}必须实现继承的抽象方法......"如果我使用快速修复,我会得到一个超过20个方法存根的怪物.
我很确定有一个更简单的解决方案,但我似乎无法找到它!
编辑:
我现在有
Type listType = new TypeToken<List<MyClass>>()
{
}.getType();
MyClass mc = new Gson().fromJson(result, listType);
Run Code Online (Sandbox Code Playgroud)
但是,我在"fromJson"行中得到以下异常:
java.lang.NullPointerException
at org.apache.harmony.luni.lang.reflect.ListOfTypes.length(ListOfTypes.java:47)
at org.apache.harmony.luni.lang.reflect.ImplForType.toString(ImplForType.java:83)
at java.lang.StringBuilder.append(StringBuilder.java:203)
at com.google.gson.JsonDeserializerExceptionWrapper.deserialize(JsonDeserializerExceptionWrapper.java:56)
at com.google.gson.JsonDeserializationVisitor.invokeCustomDeserializer(JsonDeserializationVisitor.java:88)
at com.google.gson.JsonDeserializationVisitor.visitUsingCustomHandler(JsonDeserializationVisitor.java:76)
at com.google.gson.ObjectNavigator.accept(ObjectNavigator.java:106)
at com.google.gson.JsonDeserializationContextDefault.fromJsonArray(JsonDeserializationContextDefault.java:64)
at com.google.gson.JsonDeserializationContextDefault.deserialize(JsonDeserializationContextDefault.java:49)
at com.google.gson.Gson.fromJson(Gson.java:568)
at com.google.gson.Gson.fromJson(Gson.java:515)
at com.google.gson.Gson.fromJson(Gson.java:484)
at com.google.gson.Gson.fromJson(Gson.java:434)
Run Code Online (Sandbox Code Playgroud)
我做抓JsonParseExceptions和"结果"是不为空.
我用调试器检查了listType,得到了以下内容:
我在Android应用程序中实现Json反序列化时遇到一些问题(使用Gson库)
我上过这样的课
public class MyJson<T>{
public List<T> posts;
}
Run Code Online (Sandbox Code Playgroud)
反序列化调用是:
public class JsonDownloader<T> extends AsyncTask<Void, Void, MyJson<T>> {
...
protected MyJson<T> doInBackground(Void... params) {
...
Reader reader = new InputStreamReader(content);
GsonBuilder gson = new GsonBuilder();
Type collectionType = new TypeToken<MyJson<T>>() {}.getType();
result = gson.create().fromJson(reader, collectionType);
...
}
}
Run Code Online (Sandbox Code Playgroud)
问题是调用后的result.posts列表包含一个LinkedTreeMap对象数组(具有正确的值,因此问题是反序列化)而不是MyJson对象.当我使用MyObject而不是T时,一切正常,MyObject正确.
那么有没有办法在不创建自定义反序列化器的情况下实现反序列化调用?
是否有可能在Java中反射性地实例化泛型类型?使用这里描述的技术我得到一个错误,因为类标记不能是通用的.以下面的例子为例.我想实例化一些实现Creator的Creator子类.实际的类名称作为命令行参数传入.我们的想法是能够在运行时指定Creator的实现.还有另一种方法来完成我在这里要做的事情吗?
public interface Creator<T> {
T create();
}
public class StringCreator implements Creator<String> {
public String create() { return new String(); }
}
public class FancyStringCreator implements Creator<String> {
public String create() { return new StringBuffer().toString(); }
}
public static void main(String[] args) throws Exception {
Class<?> someClass = Class.forName(args[0]);
/*ERROR*/Class<? extends Creator<String>> creatorClass = someClass.asSubclass(Creator.class);
Constructor<? extends Creator<String>> creatorCtor = creatorClass.getConstructor((Class<?>[]) null);
Creator<String> creator = creatorCtor.newInstance((Object[]) null);
}
Run Code Online (Sandbox Code Playgroud)
编辑:我喜欢马库斯的方法是最简单务实的,不会绕过整个仿制品.我可以在我的情况下使用它,因为我可以指定传递的类必须是StringCreator的子类.但是,正如Ericson指出的那样,通用信息仍然存在于类型级别,而不是在运行时级别,因此仍然可以反思地检查给定的类是否实现了正确的泛型类型.
在使用Android应用程序时,我遇到了一个问题,当我想制作一个最终的通用方法来发送HTTP请求(使用loopj)并反序列化它们(使用Gson).
您可能知道在使用gson.fromJson时不能这样做:
gson.fromJson(responseBody, new TypeToken<T>() {}.getType());
Run Code Online (Sandbox Code Playgroud)
要么
gson.fromJson(responseBody, new TypeToken<ArrayList<T>>() {}.getType())
Run Code Online (Sandbox Code Playgroud)
T您将获得LinkedTreeMap对象(或LinkedTreeMap对象列表),而不是您传递的实际对象(或第二种情况下的实际对象列表).
如果你想要仍然想要反序列化泛型类型对象(-s),请参阅下面的答案.
+ Bonus:loopj + gson的终极通用解决方案
感谢这些帖子的作者和评论者:
http://blog.xebia.com/2009/02/07/acessing-generic-types-at-runtime-in-java/