- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
以下代码片段抛出 NullPointerException。这是 Scala 的预期和正常行为吗?
object ATest extends App {
def getX[T <: X](constr: ⇒ T = null.asInstanceOf[T]): Unit = {
constr
}
getX()
}
class X
从片段生成(反编译)Java 代码:
public final class ATest {
public static void main(String[] arrstring) {
ATest$.MODULE$.main(arrstring);
}
public static void delayedInit(Function0<BoxedUnit> function0) {
ATest$.MODULE$.delayedInit(function0);
}
public static String[] args() {
return ATest$.MODULE$.args();
}
public static void scala$App$_setter_$executionStart_$eq(long l) {
ATest$.MODULE$.scala$App$_setter_$executionStart_$eq(l);
}
public static long executionStart() {
return ATest$.MODULE$.executionStart();
}
public static void delayedEndpoint$test$ATest$1() {
ATest$.MODULE$.delayedEndpoint$test$ATest$1();
}
public static <T extends X> T getX$default$1() {
return ATest$.MODULE$.getX$default$1();
}
public static <T extends X> void getX(Function0<T> function0) {
ATest$.MODULE$.getX(function0);
}
}
public final class ATest$ implements App {
public static final ATest$ MODULE$;
private final long executionStart;
private String[] scala$App$$_args;
private final ListBuffer<Function0<BoxedUnit>> scala$App$$initCode;
public static {
new test.ATest$();
}
public long executionStart() {
return this.executionStart;
}
public String[] scala$App$$_args() {
return this.scala$App$$_args;
}
public void scala$App$$_args_$eq(String[] x$1) {
this.scala$App$$_args = x$1;
}
public ListBuffer<Function0<BoxedUnit>> scala$App$$initCode() {
return this.scala$App$$initCode;
}
public void scala$App$_setter_$executionStart_$eq(long x$1) {
this.executionStart = x$1;
}
public void scala$App$_setter_$scala$App$$initCode_$eq(ListBuffer x$1) {
this.scala$App$$initCode = x$1;
}
public String[] args() {
return App.class.args((App)this);
}
public void delayedInit(Function0<BoxedUnit> body) {
App.class.delayedInit((App)this, body);
}
public void main(String[] args) {
App.class.main((App)this, (String[])args);
}
public <T extends X> void getX(Function0<T> constr) {
constr.apply();
}
public <T extends X> T getX$default$1() {
return null;
}
public final void delayedEndpoint$test$ATest$1() {
this.getX((Function0<T>)new scala.Serializable(){
public static final long serialVersionUID = 0;
public final Nothing. apply() {
return (Nothing.)ATest$.MODULE$.getX$default$1();
}
});
}
private ATest$() {
MODULE$ = this;
App.class.$init$((App)this);
this.delayedInit((Function0<BoxedUnit>)new ATest.delayedInit$body(this));
}
}
public final class ATest$.anonfun extends AbstractFunction0<Nothing.>implements Serializable {
public final Nothing. apply() {
return (Nothing.)ATest$.MODULE$.getX$default$1();
}
}
最后是 Action 部分:
public <T extends X> void getX(Function0<T> constr) {
constr.apply();
}
public <T extends X> T getX$default$1() {
return null;
}
public final void delayedEndpoint$test$ATest$1() {
this.getX((Function0<T>)new scala.Serializable(){
public final Nothing. apply() {
return (Nothing.)ATest$.MODULE$.getX$default$1();
}
});
}
即:对 getX 的调用传递了新的匿名 Function0,它的 apply() 只是调用 getX$default$1(),它是 null。所以我看不到任何可以抛出 NPE 的点。
编辑:发现 Unresolved 问题:https://issues.scala-lang.org/browse/SI-8097
编辑:表达式 null.asInstanceOf[T] 为类型 T 生成默认值。如果 Scala 将结果类型 T 推断为 Nothing,我们将进入运行时表达式
null.asInstanceOf[Nothing]
这显然将 Exeption 作为 Nothing is Exception 的默认值抛出。
但是为什么这段代码只在最后一行抛出 NPE?
object ATest extends App {
def getX[T](x: T = null.asInstanceOf[T]): T = x
getX[Nothing]() // Ok
val x = getX() // Ok
val y = null
println("x= "+x) // prints 'x= null'
println(s"y= $y") // prints 'y= null'
println(s"x= $x") // throws NPE !?
println("x==null ? "+(x==null)) // prints 'x= null'
}
为什么这个代码片段会抛出 NPE(它只是与之前的隐式参数不同)?
object ATest extends App {
def getX[T](x: T = null.asInstanceOf[T])(implicit s: String = null): T = x
getX() // throws NPE !?
}
所以情况还很模糊。问题是开放的。
最佳答案
所以我必须稍微修改一下我的答案。
然而,从字节码中可以清楚地知道是什么触发了 NPE,但在反向编译的 Java 代码中却不清楚。字节代码比 Java 代码具有更多的特性,重要的是,您可以拥有两个仅在返回类型上不同的方法并执行不同的操作。
所以让我们首先看一下堆栈跟踪:
at ATest$$anonfun$1.apply(Test.scala:7)
at ATest$.getX(Test.scala:5)
at ATest$.delayedEndpoint$ATest$1(Test.scala:7)
at ATest$delayedInit$body.apply(Test.scala:3)
at scala.Function0$class.apply$mcV$sp(Function0.scala:40)
at scala.runtime.AbstractFunction0.apply$mcV$sp(AbstractFunction0.scala:12)
at scala.App$$anonfun$main$1.apply(App.scala:76)
at scala.App$$anonfun$main$1.apply(App.scala:76)
at scala.collection.immutable.List.foreach(List.scala:383)
at scala.collection.generic.TraversableForwarder$class.foreach(TraversableForwarder.scala:35)
at scala.App$class.main(App.scala:76)
at ATest$.main(Test.scala:3)
at ATest.main(Test.scala)
所以出错的方法是 ATest$$anonfun$1.apply
让我们看看:
public final scala.runtime.Nothing$ apply();
Code:
0: getstatic #19 // Field ATest$.MODULE$:LATest$;
3: invokevirtual #23 // Method ATest$.getX$default$1:()LX;
6: checkcast #25 // class scala/runtime/Nothing$
9: areturn
public final java.lang.Object apply();
Code:
0: aload_0
1: invokevirtual #30 // Method apply:()Lscala/runtime/Nothing$;
4: athrow
首先我们注意到有两个方法叫apply,所以调用了哪个(throw是一个提示...)好吧,让我们看看调用它的方法:
public <T extends X> void getX(scala.Function0<T>);
Code:
0: aload_1
1: invokeinterface #62, 1 // InterfaceMethod scala/Function0.apply:()Ljava/lang/Object;
6: pop
7: return
因此,我们正在调用返回对象并具有抛出指令的对象。那么,为什么会出现 NullPointer 异常?
好吧,该方法执行以下操作:它将 this 放入堆栈,然后调用另一个应用方法(返回 Nothing$),该方法实际上返回 null,因为它返回我们的默认参数。现在我们在堆栈上有一个 null 并执行 throw。如果 throw 在堆栈上发现 null,则会抛出 NPE。
这就是这里发生的事情。
好吧,让我们看看在类型检查之后 scalac 是如何处理它的:
object ATest extends AnyRef with App {
def <init>(): ATest.type = {
ATest.super.<init>();
()
};
def getX[T <: X](constr: => T = null.asInstanceOf[T]): Unit = {
constr;
()
};
<synthetic> def getX$default$1[T <: X]: T = null.asInstanceOf[T];
ATest.this.getX[Nothing](ATest.this.getX$default$1[Nothing])
}
在没有 asInstanceOf 的情况下它做了什么:
object ATest extends AnyRef with App {
def <init>(): ATest.type = {
ATest.super.<init>();
()
};
def getX[T <: X](constr: => T = null): Unit = {
constr;
()
};
<synthetic> def getX$default$1[T <: X]: Null = null;
ATest.this.getX[Null](ATest.this.getX$default$1[Nothing])
}
好吧,在第一种情况下,默认参数为 Null 的信息以某种方式丢失了。
在第二种情况下,我们得到关键方法的字节码:
public final java.lang.Object apply();
Code:
0: aload_0
1: invokevirtual #27 // Method apply:()Lscala/runtime/Null$;
4: pop
5: aconst_null
6: areturn
所以在这里,编译器知道参数为 null,并使用 Null$ 类生成代码以将 null 装箱。
当然不是空指针异常。但是为什么编译器首先生成那个 throw 呢?可能是因为 asInstanceOf[T] 变成了 asInstanceOf[Nothing],如果在 null 上调用它应该抛出异常。
让我们快速尝试一下,如果我们在 repl 中这样做会发生什么:
"".asInstanceOf[Nothing]
java.lang.ClassCastException: java.lang.String cannot be cast to scala.runtime.Nothing$
到目前为止一切顺利,而且:
null.asInstanceOf[Nothing]
java.lang.NullPointerException
好吧,也许我应该从这个开始......似乎,asInstanceOf 的代码生成有一些错误并抛出错误的异常。
为什么下限 :> Null 解决了这个问题,也很清楚:推断的类型不再是 Nothing,而是 Null 并且 instanceOf 很好。
所以更有趣的问题是为什么类型检查器在您现在已删除的复杂示例上失败。
class X
object ATest extends App {
def getX[T<:X](clas: Class[T], constr: ⇒ T = null): T ={
val x = constr
if (x == null) clas.newInstance() else x
}
val clas: Class[_ <: X] = classOf[X]
getX(clas) // Ooops: type mismatch..
}
那么,类型检查器会说什么:
def getX[T <: X](clas: Class[T], constr: => T = null): T = {
val x: T = constr;
if (x.==(null))
clas.newInstance()
else
x
};
<synthetic> def getX$default$2[T <: X]: Null = null;
private[this] val clas: Class[_ <: X] = classOf[X];
<stable> <accessor> def clas: Class[_ <: X] = ATest.this.clas;
ATest.this.getX[T](<clas: error>, ATest.this.getX$default$2)
}
他无法以某种方式推断出 T 的类型,但他应该推断出 Null,因为只有引用类型的类。有趣的是,编译器并不知道这一点。好像是直接用Java中Class的定义,那里的类型参数没有下界(因为Java没有Null类型),所以下界是Nothing。这也告诉我们如何修复它:
val clas: Class[_ >: Null <: X] = classOf[X]
getX(clas)
这终于奏效了。所以你可以准确地做你一开始想做的事情。您只需告诉编译器,您对不能为空的类型的类不感兴趣。
不过,我想我还是更喜欢带有 Option 的版本:
def getX[T <: X](clas: Class[T], constr: ⇒ Option[T] = None): T = {
val x = constr
x match {
case None => clas.newInstance()
case Some(x) => x
}
}
val clas: Class[_ <: X] = classOf[X]
getX(clas)
现在也清楚了,为什么会这样:None 是一个 Option[Nothing],所以这段代码可以很好地处理 Class[Nothing]。
关于具有默认 null 的 Scala 按名称参数抛出 NullPointerException,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35245454/
我正在将我的模板代码移植到 XTend。在某些时候,我在测试用例中有这种类型的条件处理: @Test def xtendIfTest() { val obj = new FD if (
我是新来的 kotlin , 当我开始 Null Safety 时,我对下面的情况感到困惑. There's some data inconsistency with regard to initia
我的应用程序一直在各种Android版本中保持良好状态。我有用户在Android 4.3、5.0、5.1和6.0上正常运行。但是,具有S7 Edge的用户刚刚更新了Android 7.0,将文本粘贴到
我使用的是最新版本的 LWUIT (1.5)。我在资源编辑器中设计了我的表单,然后将代码生成到 netbeans。问题是如果我想访问除表单之外的任何对象,我会收到此错误: java.lang.Null
更新: 我在 Fedora 21 上运行它。 SonarQube - 5.0。 SonarQube Runner - 2.4 更新 2:Findbugs v3.1,Java 插件 v2.8 更新3:
RecupData 我的类仅在 web 中返回 NullPointerException。我连接到 pgsql db 8.3.7 - 该脚本在“控制台”syso 中运行良好 - 但引发了测试 Web
我在 mac 上使用 Processing 2.08。我正在尝试使用文档中给出的 createShape 函数创建 PShape。 PShape s; void setup(){ size(500
我在 mac 上使用 Processing 2.08。我正在尝试使用文档中给出的 createShape 函数创建 PShape。 PShape s; void setup(){ size(500
每次运行此 jsp 时,都会收到以下错误异常: org.apache.jasper.JasperException: java.lang.NullPointerException root cause
Kotlin 在编译时有一个出色的 null 检查,使用分离到“可空?”和“不可为空”的对象。它有一个 KAnnotator 来帮助确定来自 Java 的对象是否可以为空。但是,如果 not-null
我有一个布局将显示一个TextView,用于显示一个滴答时间。我遵循了此链接中的代码 How to Display current time that changes dynamically for
Elasticsearch 1.4.1版(“lucene_version”:“4.10.2”) 我有一个像这样的文件: $ curl 'http://localhost:9200/blog/artic
这是我从另一个类调用函数的方法Selenium 设置已定义。 public void Transfer() throws Exception { System.out.println("\nTrans
我试图在主类中使用我在此类中创建的函数,但它崩溃并显示“警告:无法在根 0 处打开/创建首选项根节点 Software\JavaSoft\Prefsx80000002。 Windows RegCrea
这个问题已经有答案了: What is a NullPointerException, and how do I fix it? (12 个回答) 已关闭 3 年前。 我有一个 Java 代码,它将
我声明了两张牌: Card card1 = new Card('3', Card.Suit.clubs); Card card2 = new Card('T', Card.Suit.diamonds)
我编写了一段代码来解码 Base64 图像并在 javafx 中表示该图像。在我的 url base64 代码中不断变化。这就是我在 javafx 代码中使用任务的原因。但我收到错误:java.lan
我正在尝试使用 arrayList 的 arrayList 在 Java 中实现图形。 每当调用 addEdge 函数时,我都会收到 NullPointerException 。我似乎无法弄清楚为什么
我是 Java/android 的新手,所以很多这些术语都是外国的,但我愿意学习。我不打算详细介绍该应用程序,因为我认为它不相关。我目前的问题是,我使用了博客中的教程和代码 fragment ,并使我
我正在开发一个 Android 应用程序来在 Android developer guide 的帮助下录制视频.我程序上的所有代码都与此页面相同。 我在 之外定义了权限标签。 当应
我是一名优秀的程序员,十分优秀!