我的问题很理论化...这是 Class.asSubclass ( Javadoc ) 的签名:
public <U> Class<? extends U> asSubclass(Class<U> clazz)
为什么在返回类型中使用通配符泛型?根据我对泛型的理解,更好的签名可能是:
public <U> Class<U> asSubclass(Class<U> clazz)
因为你肯定可以投
Class<? extends U>
更简单
Class<U>
Bloch 在他的书“Effective Java”中推荐(第 137 页,第 28 项):
Do not use wildcard types as return types. Rather than providing additional flexibility for your users, it would force them to use wildcard types in client code.
这个选择背后的原因是什么?我缺少什么? 非常感谢您。
编辑: 正如@egelev 所建议的那样,我确实可以用另一种方式来表达我的问题……事实上,“按原样”返回输入参数是没有意义的。所以真正的问题是: 与普通转换相比,Class.asSubclass 方法的真正用途是什么?如果出现转换问题,两者都会抛出 ClassCastException。
可能添加它是为了避免在特定情况下未经检查的强制转换:当您将 asSubclass 方法的结果直接传递给另一个要求约束类型参数的方法时,如此处(取自 Effective Java,第 146 页):
AnnotatedElement element;
...
element.getAnnotation(annotationType.asSubclass(Annotation.class));
上述方法的签名是:
<T extends Annotation> T getAnnotation(Class<T> annotationClass);
在我看来,asSubclass 方法只是一种在没有适当的编译器警告的情况下执行(实际上!)未经检查的转换的方法...
这最终又重新提出了我之前的问题:签名
public <U> Class<U> asSubclass(Class<U> clazz)
会同样有效(即使奇怪,我承认)!它将与 getAnnotation 示例完全兼容,并且不会限制客户端代码迫使它使用毫无意义的通配符泛型。
编辑2: 我想我的一般问题已经解决了;非常感谢您。如果有人有其他关于 asSubclass 签名正确性的好例子,请将它们添加到讨论中,我希望看到一个使用带有我的签名的 asSubclass 的完整示例,显然不起作用。
最佳答案
如果返回类型为Class<? extends U> .让我们先尝试理解,getClass签名:
AbstractList<String> ls = new ArrayList<>();
Class<? extends AbstractList> x = ls.getClass();
现在编译器允许我们这样做:
Class<AbstractList> x = ls.getClass();
这是错误的。因为在运行时,ls.getClass将是 ArrayList.class而不是 AbstractList.class . ls.getClass 也不能返回 Class<ArrayList>因为ls是 AbstractList<> 类型而不是 Arraylist
所以编译器现在说 - 好的!我不能返回Class<ArrayList>我也不能返回 Class<AbstractList> .但是因为我知道ls肯定是一个 AbstractList,所以实际的类对象只能是 AbstractList 的子类型。所以Class<? extends AbstractList>是一个非常安全的赌注。由于通配符:
你不能做:
AbstractList<String> ls = new ArrayList<>();
Class<? extends AbstractList> x = ls.getClass();
Class<ArrayList<?>> xx = x;
Class<AbstractList<?>> xxx = x;
同样的逻辑适用于你的问题。假设它被声明为:
public <U> Class<U> asSubClass(Class<U> c)
下面会编译:
List<String> ls = new ArrayList<>();
Class<? extends List> x = ls.getClass();
Class<AbstractList> aClass = x.asSubclass(AbstractList.class); //BIG ISSUE
以上aClass在运行时是 Class<Arraylist>而不是 Class<AbstractList> .所以这不应该被允许!! Class<? extends AbstractList>是最好的选择。
我看到这个问题的第一个想法是,为什么不将其声明为:
public <U extends T> Class<? extends U> asSubClass(Class<U> c)
对我可以传递的参数进行编译时限制更有意义。但我认为它不是首选的原因是——这会破坏 pre-java5 代码的向后兼容性。例如,如果 asSubClass,使用 pre-Java5 编译的代码将不再编译。如上所示声明。
Class x = List.class.asSubclass(String.class); //pre java5
// this would have not compiled if asSubclass was declared above like
快速检查:
public static <T, U extends T> Class<? extends U> asSubClaz(Class<T> t, Class<U> c){
return t.asSubClass(c);
}
public static <T, U> Class<? extends U> asSubClazOriginal(Class<T> t, Class<U> c){
return t.asSubClass(c);
}
asSubClazOriginal(List.class, String.class);
asSubClaz(List.class, String.class); //error. So would have broken legacy code
PS:对于编辑过的问题,关于为什么 asSubClass而不是 Actor ? - 因为 Actor 是背叛。例如:
List<String> ls = new ArrayList<>();
Class<? extends List> x = ls.getClass();
Class<AbstractList> aClass = (Class<AbstractList>) x;
上面总是会成功,因为泛型被删除了。所以它的类(class)投给了一个类(class)。但是aClass.equals(ArrayList.class)会给假的。所以肯定 Actor 是错误的。如果你需要类型安全,你可以使用 asSubClaz以上
关于java - Class.as子类签名,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/28604179/
我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co
我在开发的Rails3网站的一些搜索功能上遇到了一个小问题。我有一个简单的Post模型,如下所示:classPost我正在使用acts_as_taggable_on来更轻松地向我的帖子添加标签。当我有一个标记为“rails”的帖子并执行以下操作时,一切正常:@posts=Post.tagged_with("rails")问题是,我还想搜索帖子的标题。当我有一篇标题为“Helloworld”并标记为“rails”的帖子时,我希望能够通过搜索“hello”或“rails”来找到这篇帖子。因此,我希望标题列的LIKE语句与acts_as_taggable_on提供的tagged_with方法
我真的很习惯使用Ruby编写以下代码:my_hash={}my_hash['test']=1Java中对应的数据结构是什么? 最佳答案 HashMapmap=newHashMap();map.put("test",1);我假设? 关于java-等价于Java中的RubyHash,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/22737685/
下面例子中的Nested和Child有什么区别?是否只是同一事物的不同语法?classParentclassNested...endendclassChild 最佳答案 不,它们是不同的。嵌套:Computer之外的“Processor”类只能作为Computer::Processor访问。嵌套为内部类(namespace)提供上下文。对于ruby解释器Computer和Computer::Processor只是两个独立的类。classComputerclassProcessor#Tocreateanobjectforthisc
我正在尝试使用boilerpipe来自JRuby。我看过guide从JRuby调用Java,并成功地将它与另一个Java包一起使用,但无法弄清楚为什么同样的东西不能用于boilerpipe。我正在尝试基本上从JRuby中执行与此Java等效的操作:URLurl=newURL("http://www.example.com/some-location/index.html");Stringtext=ArticleExtractor.INSTANCE.getText(url);在JRuby中试过这个:require'java'url=java.net.URL.new("http://www
我只想对我一直在思考的这个问题有其他意见,例如我有classuser_controller和classuserclassUserattr_accessor:name,:usernameendclassUserController//dosomethingaboutanythingaboutusersend问题是我的User类中是否应该有逻辑user=User.newuser.do_something(user1)oritshouldbeuser_controller=UserController.newuser_controller.do_something(user1,user2)我
什么是ruby的rack或python的Java的wsgi?还有一个路由库。 最佳答案 来自Python标准PEP333:Bycontrast,althoughJavahasjustasmanywebapplicationframeworksavailable,Java's"servlet"APImakesitpossibleforapplicationswrittenwithanyJavawebapplicationframeworktoruninanywebserverthatsupportstheservletAPI.ht
这篇文章是继上一篇文章“Observability:从零开始创建Java微服务并监控它(一)”的续篇。在上一篇文章中,我们讲述了如何创建一个Javaweb应用,并使用Filebeat来收集应用所生成的日志。在今天的文章中,我来详述如何收集应用的指标,使用APM来监控应用并监督web服务的在线情况。源码可以在地址 https://github.com/liu-xiao-guo/java_observability 进行下载。摄入指标指标被视为可以随时更改的时间点值。当前请求的数量可以改变任何毫秒。你可能有1000个请求的峰值,然后一切都回到一个请求。这也意味着这些指标可能不准确,你还想提取最小/
HashMap中为什么引入红黑树,而不是AVL树呢1.概述开始学习这个知识点之前我们需要知道,在JDK1.8以及之前,针对HashMap有什么不同。JDK1.7的时候,HashMap的底层实现是数组+链表JDK1.8的时候,HashMap的底层实现是数组+链表+红黑树我们要思考一个问题,为什么要从链表转为红黑树呢。首先先让我们了解下链表有什么不好???2.链表上述的截图其实就是链表的结构,我们来看下链表的增删改查的时间复杂度增:因为链表不是线性结构,所以每次添加的时候,只需要移动一个节点,所以可以理解为复杂度是N(1)删:算法时间复杂度跟增保持一致查:既然是非线性结构,所以查询某一个节点的时候
遍历文件夹我们通常是使用递归进行操作,这种方式比较简单,也比较容易理解。本文为大家介绍另一种不使用递归的方式,由于没有使用递归,只用到了循环和集合,所以效率更高一些!一、使用递归遍历文件夹整体思路1、使用File封装初始目录,2、打印这个目录3、获取这个目录下所有的子文件和子目录的数组。4、遍历这个数组,取出每个File对象4-1、如果File是否是一个文件,打印4-2、否则就是一个目录,递归调用代码实现publicclassSearchFile{publicstaticvoidmain(String[]args){//初始目录Filedir=newFile("d:/Dev");Datebeg