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)

これは予想される動作ですか?

4

2 に答える 2

5

まず最初に、map ではなく foreach を使用する必要がありますが、それは既にご存知のとおりです :-)

Java で Void.TYPE に変換される Unit の配列の配列を作成しようとしています。これにより、IllegalArgumentException が発生します。

于 2011-08-08T10:57:27.503 に答える
3

Scalaは、が呼び出されたjava.lang.reflect.Array.newInstanceときに動的に配列を作成するために使用します。このメソッドは 、ゼロのサイズを受け入れませんラムダはタイプが空の配列を返しているため、サイズがゼロの場合、外部マップを呼び出すときに例外が発生します。mapArray[Unit]

ちなみに、この例では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)
...
于 2011-08-08T10:43:58.183 に答える