- Java 双重比较
- java - 比较器与 Apache BeanComparator
- Objective-C 完成 block 导致额外的方法调用?
- database - RESTful URI 是否应该公开数据库主键?
假设我有一个方法“mix”,它接受两个可能不同类型 T 和 S 的列表,并返回一个包含两者元素的列表。为了类型安全,我想指定返回的列表是 R 类型,其中 R 是 T 和 S 共有的父类(super class)型。例如:
List<Number> foo = mix(
Arrays.asList<Integer>(1, 2, 3),
Arrays.asList<Double>(1.0, 2.0, 3.0)
);
要指定这一点,我可以将方法声明为
static <R, T extends R, S extends R> List<R> mix(List<T> ts, List<S> ss)
但是如果我想制作 mix
怎么办?实例方法而不是静态方法,在类 List2<T>
上?
<R, T extends R, S extends R> List<R> mix ...
隐藏<T>
在 List2
的实例上, 所以这不好。
<R, T extends S&T, S extends R> List<R> mix ...
解决了阴影问题,但不被编译器接受
<R super T, S extends R> List<R> mix ...
被编译器拒绝,因为下界通配符不能存储在命名变量中(仅用于 ? super X
表达式)
我可以将参数移至类本身,例如 List2<R, T extends R, S extends R>
,但类型信息在实例级别确实没有任何业务,因为它仅用于一个方法调用,并且每次您想要在不同参数上调用该方法时都必须重新转换对象。
据我所知,没有办法用泛型来做到这一点。我能做的最好的就是返回一个原始的 List2
并将其转换到调用点,就像在引入泛型之前一样。有人有更好的解决方案吗?
最佳答案
如问题和评论中所述,以下签名是理想的:
<R super T, S extends R> List<R> mix(List<S> otherList)
当然,R super T
is not allowed by the language (请注意,polygenelubricants 在链接帖子上的回答是错误的 - 正如您的问题所示,此语法有一些用例)。
这里没有办法取胜 - 您只有几种解决方法中的一种可供选择:
mix
静态方法。这实际上是一个不错的选择,除非出于与多态性相关的原因它需要成为类接口(interface)的一部分,或者您计划 mix
如此常用的方法,以至于您认为保持静态是 Not Acceptable 。mix
的签名解决过于严格,并记录调用者需要进行某些未经检查的强制转换。这类似于 Guava的 Optional.or
不得不做。来自该方法的文档:Note about generics: The signature
public T or(T defaultValue)
is overly restrictive. However, the ideal signature,public <S super T> S or(S)
, is not legal Java. As a result, some sensible operations involving subtypes are compile errors:Optional<Integer> optionalInt = getSomeOptionalInt();
Number value = optionalInt.or(0.5); // errorAs a workaround, it is always safe to cast an
Optional<? extends T>
toOptional<T>
. Casting [the aboveOptional
instance] toOptional<Number>
(whereNumber
is the desired output type) solves the problem:Optional<Number> optionalInt = (Optional) getSomeOptionalInt();
Number value = optionalInt.or(0.5); // fine
对你来说不幸的是,并不总是安全地施放List2<? extends T>
至 List2<T>
.例如,类型转换 List2<Integer>
到 List2<Number>
可以允许 Double
添加到本应容纳的东西中 Integer
s 并导致意外的运行时错误。异常(exception)情况是 List2
是不可变的(如 Optional
),但这似乎不太可能。
不过,如果您小心并记录类型不安全的代码并附上解释,您仍然可以摆脱此类强制转换。假设 mix
具有以下签名(和实现,为了好玩):
List<T> mix(final List<? extends T> otherList) {
final int totalElements = (size() + otherList.size());
final List<T> result = new ArrayList<>(totalElements);
Iterator<? extends T> itr1 = iterator();
Iterator<? extends T> itr2 = otherList.iterator();
while (result.size() < totalElements) {
final T next = (itr1.hasNext() ? itr1 : itr2).next();
result.add(next);
final Iterator<? extends T> temp = itr1;
itr1 = itr2;
itr2 = temp;
}
return result;
}
那么你可能有以下调用站点:
final List2<Integer> ints = new List2<>(Arrays.asList(1, 2, 3));
final List<Double> doubles = Arrays.asList(1.5, 2.5, 3.5);
final List<Number> mixed;
// type-unsafe code within this scope
{
@SuppressWarnings("unchecked") // okay because intsAsNumbers isn't written to
final List2<Number> intsAsNumbers = (List2<Number>)(List2<?>)ints;
mixed = intsAsNumbers.mix(doubles);
}
System.out.println(mixed); // [1, 1.5, 2, 2.5, 3, 3.5]
同样,解决静态问题 mix
将变得更清洁并且没有类型安全风险。我会确保有充分的理由不保持这种状态。
关于java - 在 Java 中使用泛型存储公共(public)父类(super class)型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18303724/
这个问题在这里已经有了答案: 关闭 10 年前。 Possible Duplicate: When should [assembly: InternalsVisibleTo()] be used?
问题与微服务有关,当我有多个微服务提供将被订购和计费的功能/服务时。 我正在确定采用哪种方法, a) 每个可计费微服务有一个订单和一个计费服务,有各自的数据库。b) 跨所有微服务的通用订单管理和计费服
我正在尝试使用 gcloud图书馆。 (ns firengine.state (:import [com.google.cloud AuthCredentials] [com.goog
Java 允许定义以下一对类。 class Class1 { ... } public Class2 { public Class2(Class1 c1) { ... } } 如果因为 Class1
我正在尝试查找文件 1 和文件 2 中的共同行。如果公共(public)行存在,我想写入文件 2 中的行,否则打印文件 1 中的非公共(public)行。fin1 和 fin2 是这里的文件句柄。它读
好吧,这是一个满口的标题。不过,这让我明白了。这是我的代码的要点,在 jar 里: public class NetworkShared { public static class Login
我在使用 ltree 时遇到 PHP 问题来自 PostgreSQL .我在 SQL 中这样做: SELECT * FROM tabla t WHERE t.parent_path " for "OP
我知道如何为类/接口(interface)/包的子集生成 Javadoc。但是有没有办法只为公共(public)方法的一个子集生成 Javadoc? 我更喜欢能够将方法(Javadoc 标记或注释)标
这个问题在这里已经有了答案: 关闭 12 年前。 Possible Duplicates: c#: why have empty get set properties instead of usin
在我们的每个项目中,都有一个文件用于存储该项目中使用的各种SQL 语句。类的声明方式和字符串的声明方式有一些变化。 示例类声明: internal sealed class ClassName int
我根据 http://docs.jquery.com/Plugins/Authoring 定义了我的插件 (function( $ ){ var methods = { init : fu
我正在使用 Inno Setup 来构建我的安装程序,我有 C:\Users\Public文件夹硬编码在我的 [Files] 中放置一些文件的部分(Inno Setup 没有此文件夹的常量) 我的目标
我有一个 dataframe1 包含像 'ID', 'A', 'B', 'C', 'D', 'E', 'F', 'G' 这样的列. 现在,我创建了两个数据框, dataframe2 包含 'ID',
我有一个抽象类,不幸的是我无法更改它的定义,它基本上提供了一个抽象方法,有点像。 public abstract void do(Data someData, BaseInterface interf
我刚刚在重构时偶然发现了一段奇怪的代码。它看起来像是分解出两个 readString() 方法的共同部分的候选者,只是它似乎是不可能的(这对我来说是一个令人毛骨悚然的脑筋急转弯): private f
是否有解析为公用文件夹的属性?显然,我不想在目录结构中对“c:\users\public”进行硬编码,但我找不到预定义的 Property解决这个问题。是否有一种可接受的方式来指定要在此处安装和/或在
我试图将值从一个类传递到另一个类。 subPanel1 类读取全局变量,但当我通过调整监听器更新这些变量时,它不会更改值。我试图将 rc、gc 和 bc 变量从 subPanel2 类传递到 subP
我想使用具有自动属性的干净且编码较少的类。所有属性(property)都是公共(public)的。在同一类的方法中我也使用了该属性。因此,我认为这种方法是可混搭的,因为我将公共(public)属性用于
不久前,我在 Android 应用程序中创建了一个 SQLiteHelper 类。我不是 100% 确定原因,但表名和列名是嵌套公共(public)静态抽象类中的公共(public)静态最终字段。我记
这个问题已经有答案了: Cannot make a static reference to the non-static method (8 个回答) 已关闭 3 年前。 我正在为类(class)做一
我是一名优秀的程序员,十分优秀!