- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
为什么只有在 Main.main() 中取消注释第三条语句时才会出现 ClassCastException ?没有异常(exception),但第一条和第二条语句执行得很好?
public class Tuple<K, V> {
public final K first;
public final V second;
public Tuple(K first, V second) {
this.first = first;
this.second = second;
}
@Override public String toString() {
return "Tuple{" + "first = " + first + ", second = " + second + '}';
}
}
class Test { static Tuple f(){return new Tuple("test", 8);} }
class Bar {}
class Main{
public static void main(String[] args) {
Tuple<String, Bar> t = Test.f();
System.out.println(t);
//System.out.println(t.second.getClass().getSimpleName());
}
}
提前致谢。
最佳答案
当您编写方法调用链时:
System.out.println(t.second.getClass().getSimpleName());
编译器有效地将其扩展为:
TypeOfTSecond tmpTSecond = t.second;
Class<?> clazzTmp = tmp.getClass();
String nameTmp = clazzTmp.getSimpleName();
System.out.println(nameTmp);
现在,如果 t.second
碰巧是泛型类型,编译器将插入一个转换为 t.second
将是的类型:
Bar tmpTSecond = (Bar) t.second;
因此,即使您从未访问任何 Bar
特定的功能,您也会收到 ClassCastException
。
为了演示这一点,这里是字节码:
public static void main(java.lang.String[]);
Code:
0: invokestatic #2 // Method Test.f:()LTuple;
3: astore_1
4: getstatic #3 // Field java/lang/System.out:Ljava/io/PrintStream;
7: aload_1
8: getfield #4 // Field Tuple.second:Ljava/lang/Object;
11: checkcast #5 // class Bar
14: invokevirtual #6 // Method java/lang/Object.getClass:()Ljava/lang/Class;
17: invokevirtual #7 // Method java/lang/Class.getSimpleName:()Ljava/lang/String;
20: invokevirtual #8 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
23: return
第 8 行是将 t.second
压入堆栈的位置;第 11 行是转换为 Bar
的地方。
这只是因为声明 test.f()
时使用的原始类型而发生:
static Tuple f(){return new Tuple("test", 8);}
如果这被正确声明为
static Tuple<String, Integer> f(){return new Tuple<>("test", 8);}
然后这一行
Tuple<String, Bar> t = Test.f();
无法编译。但使用原始类型会禁用编译器的类型检查,因此不能保证防止此类运行时错误。
<小时/>主要的外卖类(class)是 never use raw types .
第二个教训是注意编译器(或 IDE)的警告。编译这段代码时,我被告知:
Note: Main.java uses unchecked or unsafe operations.
Note: Recompile with -Xlint:unchecked for details.
当我使用该标志重新编译时:
Main.java:19: warning: [unchecked] unchecked call to Tuple(K,V) as a member of the raw type Tuple
return new Tuple("test", 8);
^
where K,V are type-variables:
K extends Object declared in class Tuple
V extends Object declared in class Tuple
Main.java:26: warning: [unchecked] unchecked conversion
Tuple<String, Bar> t = Test.f();
^
required: Tuple<String,Bar>
found: Tuple
2 warnings
关于Java 泛型 "upcast"到非参数化类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38775411/
我有这个例子: class One { public void testOne(){System.out.println("One!!!");} public void testTwo
我可以看到“upcast”一词与OOP有关,但我通过搜索互联网找不到确切的定义。 谁能解释这个术语是什么意思以及这种技术在什么情况下有用? 最佳答案 根据您发布的标签的描述: Upcasting pe
我定义了一个带有抽象约束的抽象类: abstract class Asset where P : Parm { } abstract class Parm { } class StockParm :
假设B是D的基类(可能是虚的,可能是多重继承,不一定是直接基类)。 设 obj 是 D 类型的对象(不是 D 的子类——恰好是 D ). 让 D * d = std::addressof(obj);
假设我有一个可以保存基类方法地址的指针类型。我可以将子类方法的地址分配给它并期望它正常工作吗?在我的例子中,我将它与基类指针一起使用,对象的动态类型是派生类。 struct B { typed
抽象类的目的不是让开发人员创建基类的对象然后向上转换它,AFAIK。 现在,即使不需要向上转换,我仍然使用它,它是否在某些方面被证明是“不利”? 更多说明: 来自 C++ 中的思考: Often in
为什么只有在 Main.main() 中取消注释第三条语句时才会出现 ClassCastException ?没有异常(exception),但第一条和第二条语句执行得很好? public class
只是玩类型转换。假设,我们有 2 个类 public class Base { public int a; } public class Inh : Base { public int
我有以下java代码: class A { int someMethod () { return 1; } int someMethod (A a) { return 2; } int some
两个类: public class ClassA implements ClassC { private static final long serialVersionUID = 111111
在 Java 中, "Up-casting is casting to a supertype, while downcasting is casting to a subtype. Supercas
我有 2 个类 Test(Base) 和 Program(child)。现在我在沮丧的时候遇到了一些问题。 Test t = new Program();// upcasting-wo
如果我有一个带有签名(String, Bool) 的元组,我不能将它转换为(String, Any)。编译器说: error: cannot express tuple conversion '(St
C# 工厂模式是否需要向上转型? 我希望类库 G 中的上帝在类库 A 中创建一个亚当,而不是让 G 依赖于 A。上帝产生亚当供类库 E 中的夏娃消费,夏娃知道并依赖亚当是可以的。(编辑 - 这个示例越
class A { } class B extends A { } class Main { public static void main(String[] args) {
假设我有一个现有的聚合 CustomAggregate 和一个由该聚合处理的事件 CustomEvent。 一段时间后,当事件存储已包含此聚合的一些事件时,我需要将聚合和事件重命名为 NewCusto
我正在使用 FunKTiale 库执行模式匹配。然后,当我按如下方式初始化变量时: private lateinit var socket = Option.None 下面的转换是不可能的: sock
我有两个 View ,A 和 B,都继承自 C。 在 Storyboard 中,我将 UIViewController 定义为 C 类型。 另一个 View ,比方说 D,有到 C 的两个 segue
我有一个二维 numpy 数组,我想将特定的 dtype 应用于每一列。 a = np.arange(25).reshape((5,5)) In [40]: a Out[40]: array([[ 0
我一直在阅读该站点上的其他一些主题,他们提到了 dynamic_cast 和 static_cast 是如何安全地进行向上转换的。 为什么向上转换甚至需要这些? 例如,如果类 B 派生自 A,则 A
我是一名优秀的程序员,十分优秀!