有人能解释一下为什么在Scala中这样应用map会抛出IllegalArgumentException异常吗?

3
以下是解释器命令的副本:
scala>val myTable = Array(Array(1))
res30: Array[Array[Int]] = Array(Array(1))

scala> myTable.map(_.map(_.toString))
res31: Array[Array[java.lang.String]] = Array(Array(1))

scala> var result = 0
result: Int = 0

scala> myTable.head
res32: Array[Int] = Array(1)

//note how applying this works
scala> res32.map(elem => if(3> result) result = 3)
res34: Array[Unit] = Array(())

scala> result
res35: Int = 3

//this also works
scala> myTable.map(_.map(_.toString))
res31: Array[Array[java.lang.String]] = Array(Array(1))

//when you combine the double map application and the anonymous function from earlier:
scala> myTable.map(_.map(elem => if(3 > result) result = 3))
java.lang.IllegalArgumentException
    at java.lang.reflect.Array.newArray(Native Method)
    at java.lang.reflect.Array.newInstance(Array.java:52)
    at scala.reflect.ClassManifest$class.arrayClass(ClassManifest.scala:107)
    at scala.reflect.Manifest$$anon$9.arrayClass(Manifest.scala:152)
    at scala.reflect.Manifest$class.arrayManifest(Manifest.scala:46)
    at scala.reflect.Manifest$$anon$9.arrayManifest(Manifest.scala:152)
    at scala.reflect.Manifest$$anon$9.arrayManifest(Manifest.scala:152)
    at scala.reflect.ClassManifest$.arrayType(ClassManifest.scala:205)
    at .<init>(<console>:45)
    at .<clinit>(<console>)
at .<init>(<console>:11)
at .<clinit>(<console>)
at $export(<console>)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:39)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:25)
at java.lang.reflect.Method.invoke(Method.java:597)
at scala.tools.nsc.interpreter.IMain$ReadEvalPrint.call(IMain.scala:592)
at scala.tools.nsc.interpreter.IMain$Request$$anonfun$10.apply(IMain.scala:828)
at scala.tools.nsc.interpreter.Line$$anonfun$1.apply$mcV$sp(Line.scala:43)
at scala.tools.nsc.io.package$$anon$2.run(package.scala:31)
at java.lang.Thread.run(Thread.java:662)

这是预期行为吗?
2个回答

5

首先,你应该使用foreach而不是map,但你已经知道了这一点:-)

你正在尝试创建一个Unit数组的数组,它在Java中被翻译为Void.TYPE。这会导致IllegalArgumentException异常。


3
Scala使用java.lang.reflect.Array.newInstance在调用map时动态创建数组。由于您的lambda返回一个类型为Array[Unit]的空数组,其大小为零,因此当调用外部map时会出现异常。请注意,此方法不接受大小为零的数组

顺便说一句,在这个例子中,您应该使用foreach,因为您并不是要将数组映射到另一个数组,而是要使用副作用:

scala> myTable.foreach(_.foreach(elem => if(3 > result) result = 3))

scala> result
res21: Int = 3

编辑:我误解了问题(请参见MatthieuF的答案)。问题不在于数组的大小,而在于类型。只需使用以下代码即可重现问题:

scala> java.lang.reflect.Array.newInstance(classOf[Unit], 1)
java.lang.IllegalArgumentException
at java.lang.reflect.Array.newArray(Native Method)
at java.lang.reflect.Array.newInstance(Array.java:52)
at .<init>(<console>:8)
...

是的,它可以接受大小为零。使用的方法是:public static Object newInstance(Class<?> componentType, int length) throws NegativeArraySizeException。不是另一个方法。 - Matthew Farwell
尝试以下代码:java.lang.reflect.Array.newInstance(int.class, 0) - Matthew Farwell

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接