Scala 集合:完全不可预测的行为
对 Scala 2.8 的收集行为感到非常沮丧。问题是:我正在创建一个数独板。我正在标记从 A1 到 I9 的单元格(字母是行,数字是列)。我想获得棋盘上的单位列表,即9行、夜间列和夜间象限。
这是我的 scala 类:
class Square(val row:Char, val column:Int) extends Pair[Char, Int](row, column) {
override def toString() = "" + row + column
}
object Board {
private val rows = "ABCDEFGHI"
private val cols = 1 to 9
private lazy val units = unitList(rows, cols)
private def cross(rows:Iterable[Char], columns:Iterable[Int]):Iterable[Square] = {
for (row <- rows; col <- columns)
yield new Square(row, col)
}
private def unitList(rows:Iterable[Char], cols:Iterable[Int]) = {
val u1 = (for (col <- cols) yield cross(rows, List(col)))
val u2 = (for (row <- rows) yield cross(List(row), cols))
val u3 = (for (cols <- List("ABC", "DEF", "GHI"); rows <- List(1 to 3, 4 to 6, 7 to 9)) yield cross(cols, rows))
u1 :+ u2 :+ u3 // won't compile, reason: :+ is not a member of Iterable[Iterable[sudoku.Square]]
}
def run() {
val u1 = (for (col <- cols) yield cross(rows, List(col)))
val u2 = (for (row <- rows) yield cross(List(row), cols))
val u3 = (for (cols <- List("ABC", "DEF", "GHI"); rows <- List(1 to 3, 4 to 6, 7 to 9)) yield cross(cols, rows))
println(u1)
println(u2)
println(u3)
val u4 = u1 :+ u2 :+ u3 // compiles
println(u1 :+ u2 :+ u3) // compiles and output correctly
}
}
请参阅代码中的注释。具体来说,为什么相同的代码不会在unitList中编译,但在run()中编译并运行良好?
另外,当我观察 run 方法的输出时,yield 关键字返回的集合似乎在 Vector 和 List 之间随机切换:
Vector(Vector(A1, B1, C1, D1, E1, F1, G1, H1, I1), Vector(A2, B2, C2, D2, E2, F2, G2, H2, I2), Vector(A3, B3, C3, D3, E3, F3, G3, H3, I3), Vector(A4, B4, C4, D4, E4, F4, G4, H4, I4), Vector(A5, B5, C5, D5, E5, F5, G5, H5, I5), Vector(A6, B6, C6, D6, E6, F6, G6, H6, I6), Vector(A7, B7, C7, D7, E7, F7, G7, H7, I7), Vector(A8, B8, C8, D8, E8, F8, G8, H8, I8), Vector(A9, B9, C9, D9, E9, F9, G9, H9, I9))
Vector(List(A1, A2, A3, A4, A5, A6, A7, A8, A9), List(B1, B2, B3, B4, B5, B6, B7, B8, B9), List(C1, C2, C3, C4, C5, C6, C7, C8, C9), List(D1, D2, D3, D4, D5, D6, D7, D8, D9), List(E1, E2, E3, E4, E5, E6, E7, E8, E9), List(F1, F2, F3, F4, F5, F6, F7, F8, F9), List(G1, G2, G3, G4, G5, G6, G7, G8, G9), List(H1, H2, H3, H4, H5, H6, H7, H8, H9), List(I1, I2, I3, I4, I5, I6, I7, I8, I9))
List(Vector(A1, A2, A3, B1, B2, B3, C1, C2, C3), Vector(A4, A5, A6, B4, B5, B6, C4, C5, C6), Vector(A7, A8, A9, B7, B8, B9, C7, C8, C9), Vector(D1, D2, D3, E1, E2, E3, F1, F2, F3), Vector(D4, D5, D6, E4, E5, E6, F4, F5, F6), Vector(D7, D8, D9, E7, E8, E9, F7, F8, F9), Vector(G1, G2, G3, H1, H2, H3, I1, I2, I3), Vector(G4, G5, G6, H4, H5, H6, I4, I5, I6), Vector(G7, G8, G9, H7, H8, H9, I7, I8, I9))
Vector(Vector(A1, B1, C1, D1, E1, F1, G1, H1, I1), Vector(A2, B2, C2, D2, E2, F2, G2, H2, I2), Vector(A3, B3, C3, D3, E3, F3, G3, H3, I3), Vector(A4, B4, C4, D4, E4, F4, G4, H4, I4), Vector(A5, B5, C5, D5, E5, F5, G5, H5, I5), Vector(A6, B6, C6, D6, E6, F6, G6, H6, I6), Vector(A7, B7, C7, D7, E7, F7, G7, H7, I7), Vector(A8, B8, C8, D8, E8, F8, G8, H8, I8), Vector(A9, B9, C9, D9, E9, F9, G9, H9, I9), Vector(List(A1, A2, A3, A4, A5, A6, A7, A8, A9), List(B1, B2, B3, B4, B5, B6, B7, B8, B9), List(C1, C2, C3, C4, C5, C6, C7, C8, C9), List(D1, D2, D3, D4, D5, D6, D7, D8, D9), List(E1, E2, E3, E4, E5, E6, E7, E8, E9), List(F1, F2, F3, F4, F5, F6, F7, F8, F9), List(G1, G2, G3, G4, G5, G6, G7, G8, G9), List(H1, H2, H3, H4, H5, H6, H7, H8, H9), List(I1, I2, I3, I4, I5, I6, I7, I8, I9)), List(Vector(A1, A2, A3, B1, B2, B3, C1, C2, C3), Vector(A4, A5, A6, B4, B5, B6, C4, C5, C6), Vector(A7, A8, A9, B7, B8, B9, C7, C8, C9), Vector(D1, D2, D3, E1, E2, E3, F1, F2, F3), Vector(D4, D5, D6, E4, E5, E6, F4, F5, F6), Vector(D7, D8, D9, E7, E8, E9, F7, F8, F9), Vector(G1, G2, G3, H1, H2, H3, I1, I2, I3), Vector(G4, G5, G6, H4, H5, H6, I4, I5, I6), Vector(G7, G8, G9, H7, H8, H9, I7, I8, I9)))
我在这里完全迷失了。
Have been pretty frustrated by Scala 2.8 collection behaviours. Here's the problem: I'm creating a Sudoku board. I'm labelling the cells from A1 to I9 (the letters being the rows and the digits being the columns). I want to get a list of units on the board, which is the 9 rows, the night columns, and the night quadrants.
Here's my scala class:
class Square(val row:Char, val column:Int) extends Pair[Char, Int](row, column) {
override def toString() = "" + row + column
}
object Board {
private val rows = "ABCDEFGHI"
private val cols = 1 to 9
private lazy val units = unitList(rows, cols)
private def cross(rows:Iterable[Char], columns:Iterable[Int]):Iterable[Square] = {
for (row <- rows; col <- columns)
yield new Square(row, col)
}
private def unitList(rows:Iterable[Char], cols:Iterable[Int]) = {
val u1 = (for (col <- cols) yield cross(rows, List(col)))
val u2 = (for (row <- rows) yield cross(List(row), cols))
val u3 = (for (cols <- List("ABC", "DEF", "GHI"); rows <- List(1 to 3, 4 to 6, 7 to 9)) yield cross(cols, rows))
u1 :+ u2 :+ u3 // won't compile, reason: :+ is not a member of Iterable[Iterable[sudoku.Square]]
}
def run() {
val u1 = (for (col <- cols) yield cross(rows, List(col)))
val u2 = (for (row <- rows) yield cross(List(row), cols))
val u3 = (for (cols <- List("ABC", "DEF", "GHI"); rows <- List(1 to 3, 4 to 6, 7 to 9)) yield cross(cols, rows))
println(u1)
println(u2)
println(u3)
val u4 = u1 :+ u2 :+ u3 // compiles
println(u1 :+ u2 :+ u3) // compiles and output correctly
}
}
See the comments in the code. Specifically, why the same code won't compile in unitList but compiles and runs fine in run()?
Also, when I observe the output of the run method, it seems the collection returned by the yield keyword is randomly switching between Vector and List:
Vector(Vector(A1, B1, C1, D1, E1, F1, G1, H1, I1), Vector(A2, B2, C2, D2, E2, F2, G2, H2, I2), Vector(A3, B3, C3, D3, E3, F3, G3, H3, I3), Vector(A4, B4, C4, D4, E4, F4, G4, H4, I4), Vector(A5, B5, C5, D5, E5, F5, G5, H5, I5), Vector(A6, B6, C6, D6, E6, F6, G6, H6, I6), Vector(A7, B7, C7, D7, E7, F7, G7, H7, I7), Vector(A8, B8, C8, D8, E8, F8, G8, H8, I8), Vector(A9, B9, C9, D9, E9, F9, G9, H9, I9))
Vector(List(A1, A2, A3, A4, A5, A6, A7, A8, A9), List(B1, B2, B3, B4, B5, B6, B7, B8, B9), List(C1, C2, C3, C4, C5, C6, C7, C8, C9), List(D1, D2, D3, D4, D5, D6, D7, D8, D9), List(E1, E2, E3, E4, E5, E6, E7, E8, E9), List(F1, F2, F3, F4, F5, F6, F7, F8, F9), List(G1, G2, G3, G4, G5, G6, G7, G8, G9), List(H1, H2, H3, H4, H5, H6, H7, H8, H9), List(I1, I2, I3, I4, I5, I6, I7, I8, I9))
List(Vector(A1, A2, A3, B1, B2, B3, C1, C2, C3), Vector(A4, A5, A6, B4, B5, B6, C4, C5, C6), Vector(A7, A8, A9, B7, B8, B9, C7, C8, C9), Vector(D1, D2, D3, E1, E2, E3, F1, F2, F3), Vector(D4, D5, D6, E4, E5, E6, F4, F5, F6), Vector(D7, D8, D9, E7, E8, E9, F7, F8, F9), Vector(G1, G2, G3, H1, H2, H3, I1, I2, I3), Vector(G4, G5, G6, H4, H5, H6, I4, I5, I6), Vector(G7, G8, G9, H7, H8, H9, I7, I8, I9))
Vector(Vector(A1, B1, C1, D1, E1, F1, G1, H1, I1), Vector(A2, B2, C2, D2, E2, F2, G2, H2, I2), Vector(A3, B3, C3, D3, E3, F3, G3, H3, I3), Vector(A4, B4, C4, D4, E4, F4, G4, H4, I4), Vector(A5, B5, C5, D5, E5, F5, G5, H5, I5), Vector(A6, B6, C6, D6, E6, F6, G6, H6, I6), Vector(A7, B7, C7, D7, E7, F7, G7, H7, I7), Vector(A8, B8, C8, D8, E8, F8, G8, H8, I8), Vector(A9, B9, C9, D9, E9, F9, G9, H9, I9), Vector(List(A1, A2, A3, A4, A5, A6, A7, A8, A9), List(B1, B2, B3, B4, B5, B6, B7, B8, B9), List(C1, C2, C3, C4, C5, C6, C7, C8, C9), List(D1, D2, D3, D4, D5, D6, D7, D8, D9), List(E1, E2, E3, E4, E5, E6, E7, E8, E9), List(F1, F2, F3, F4, F5, F6, F7, F8, F9), List(G1, G2, G3, G4, G5, G6, G7, G8, G9), List(H1, H2, H3, H4, H5, H6, H7, H8, H9), List(I1, I2, I3, I4, I5, I6, I7, I8, I9)), List(Vector(A1, A2, A3, B1, B2, B3, C1, C2, C3), Vector(A4, A5, A6, B4, B5, B6, C4, C5, C6), Vector(A7, A8, A9, B7, B8, B9, C7, C8, C9), Vector(D1, D2, D3, E1, E2, E3, F1, F2, F3), Vector(D4, D5, D6, E4, E5, E6, F4, F5, F6), Vector(D7, D8, D9, E7, E8, E9, F7, F8, F9), Vector(G1, G2, G3, H1, H2, H3, I1, I2, I3), Vector(G4, G5, G6, H4, H5, H6, I4, I5, I6), Vector(G7, G8, G9, H7, H8, H9, I7, I8, I9)))
I'm totally at lost here.
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(3)
for 理解中的产量结果是从第一个生成器的类型派生的,在您的方法中,您限制了参数的类型,到目前为止您丢失了 :+ 方法。
The result of the yield in the for comprehension is derived from the type of the first generator, in you method, you're constraining the types of your parameters so far you lose your :+ methods.
尽管行为确实很奇怪,但
:+
不可能是您想要的运算符。由于您没有注释unitList
的返回类型,我不知道您期望什么。我假设您想要返回Iterable[Square]
或Iterable[Iterable[Square]]
。那么让我们看看如何获取它们,以及为什么:+
不正确。首先,
u1
、u2
和u3
都是Iterable[Iterable[Square]]
,尽管确切的亚型各异。这应该很容易理解:cross
返回Iterable[Square]
,因此在 for 理解中产生cross
会产生Iterable [可迭代[方形]]
。接下来,让我们考虑
:+
。此方法向集合添加一个元素,因此,如果u1
是Iterable(a, b, c)
,其中 a、b 和 c 是Iterable[Square ]
,则u1 :+ u2
为Iterable(a, b, c, u2)
,其类型变为Iterable[X]
code>,其中X
是Iterable[Square]
(a、b 和 c 的类型)和Iterable[Iterable[Square]]
的统一code>(u2
的类型)。最终结果是一个Iterable[Iterable[AnyRef]]
。由于
u1
、u2
和u3
的类型本质上是相同的,因此正确的操作很可能是这样的:它将返回
可迭代[可迭代[方形]]
。现在,如果您想删除嵌套并返回 Iterable[Square],您可以将其展平:这两件事之一可能就是您想要的。
现在,至于“随机”切换,没有什么随机的。在每种情况下,都有两个 for 推导式,并且结果集合的实际实现取决于原始集合的实现。因此,让我们考虑一下:
Range
,内部类型源自String
(要交叉的第一个参数)String
code>,内部类型来自List
(要交叉的第一个参数)List
,内部类型源自String
(第一个参数 <因此可以很容易地推断出对
String
(实际上是WrappedString
)和Range
的 for 理解会产生Vector /code>,而对
List
的 for 推导则生成List
。Notwithstanding the really weird behavior,
:+
cannot possibly be the operator you want. Since you did not annotate the return type ofunitList
, I don't know what you expected. I assume you wanted to return eitherIterable[Square]
orIterable[Iterable[Square]]
. So let's see how you can get them, and why:+
is incorrect.First of all,
u1
,u2
andu3
are allIterable[Iterable[Square]]
, though the exact subtype varies. This should be easy to understand:cross
returnsIterable[Square]
, so yieldingcross
in a for-comprehension results in anIterable[Iterable[Square]]
.Next, Let's consider
:+
. This method adds an element to a collection, so, ifu1
isIterable(a, b, c)
, where a, b and c areIterable[Square]
, thenu1 :+ u2
isIterable(a, b, c, u2)
, and its type becomesIterable[X]
, whereX
is the unification ofIterable[Square]
(the type of a, b and c) andIterable[Iterable[Square]]
(the type ofu2
). The end result is anIterable[Iterable[AnyRef]]
.Since the type of
u1
,u2
andu3
are essentially the same, the correct operation in all likelyhood is this:Which will return
Iterable[Iterable[Square]]
. Now, if you want to remove the nesting and returnIterable[Square]
, you can flatten this:One of these two things is probably what you want.
Now, as for the "random" switching, there's nothing random about it. In each case, there are two for-comprehensions, and the actual implementation of the resulting collection depends on the implementation of the original collection. So, let's consider it:
Range
, inner type fromString
(first parameter to cross)String
, inner type fromList
(first parameter to cross)List
, inner type derives fromString
(first parameter to cross)So it can be easily deduced that for-comprehensions over
String
(WrappedString
, actually) andRange
result inVector
, while for-comprehensions overList
result inList
.发生编译器错误是因为
:+
是IndexedSeq
的成员(因此也是List
)的成员,但不是Iterable
的成员。如果将unitList
的返回值更改为 ,它就
可以正常编译。
The compiler error occurs because
:+
is a member ofIndexedSeq
(and thusList
) but not ofIterable
. If you change the return value ofunitList
fromto
it compiles just fine.