2

作为一个 Scala 初学者,我试图为表达式Iterator中检索和处理的每个项目实现一个计数器,并且每次在for表达式的一个“循环”(外循环和嵌套循环)开始。counter = counter + 1要求是在不简单地在表达式中的多个位置放置语句的情况下完成此操作for。下面的清单显示了我对这个问题提出的解决方案,我想知道,为什么方法next实现Iterator的抽象成员被调用(并且相应的计数器增加),而flatMap覆盖map它们在 trait 中定义的挂件Iterator(并通过调用它们super)不是根本就叫。

object ZebraPuzzle {
  var starts = 0
  var items = 0

  class InstrumentedIter[A](it: Iterator[A]) extends Iterator[A] {
    private val iterator = it

    def hasNext = it.hasNext

    def next() = {
      items = items + 1
      it.next()
    }

    override def flatMap[B](f: (A) ⇒ GenTraversableOnce[B]): Iterator[B] = {
      starts = starts + 1
      super.flatMap(f)
    }

    override def map[B](f: (A) ⇒ B): Iterator[B] = {
      starts = starts + 1
      super.map(f)
    }
  } // inner class InstrumentedIter 

对应的for表达式如下所示:

  def solve = {
    val first = 1
    val middle = 3
    val houses = List(first, 2, middle, 4, 5)
    for {
      List(r, g, i, y, b) <-  new InstrumentedIter(houses.permutations)
      if ...
      List(eng, span, ukr, jap, nor) <- new InstrumentedIter(houses.permutations)
      if ...
      if ...
      if ...
      List(of, tea, milk, oj, wat) <- new InstrumentedIter(houses.permutations)
      if ...
      ...
    } yield ...
  ...
  }
...
} // standalone singleton object ZebraPuzzle

如果有人能给我提示如何以更好的方式解决给定的问题,我将不胜感激。但最重要的是,我很想知道为什么我的解决方案会覆盖Iterator'smap并且flatMap不像我有限的大脑所期望的那样工作;-)

问候

马丁

4

2 回答 2

0

与此同时,我设法自己找到了答案。我的解决方案的问题是 withFilter 返回对新创建的 AbstractIterator 的引用,而不是 InstrumentedIterator。作为一种可能的解决方案,可以将此引用传递给包装类的构造函数,例如 InstrumentedIterator,它混合了 trait Iterator 并覆盖方法 map 和 flatMap。然后这些方法可以进行计数...

问候马丁

于 2015-04-27T21:50:50.783 回答
-1

你的台词

List(...) <- Iterator

不要调用地图和平面图。他们在 List 伴生对象中调用 unapply,将迭代器解包成元组。

要调用 map 或 flatMap 你需要类似的东西

item <- Iterator

您需要使用 unapply 方法为 InstrumentedIter 定义一个伴随对象,或者在您的理解中使用 map/flatMap 语法。

于 2015-04-18T14:03:26.410 回答