Scala List.filter with two conditions, applied only once

前端 未结 4 374
旧时难觅i
旧时难觅i 2021-02-04 13:18

Don\'t know if this is possible, but I have some code like this:

val list = List(1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12)
val evens = list.filter { e => e % 2 =         


        
4条回答
  •  旧时难觅i
    2021-02-04 14:17

    If you turn list into a lazy collection, such as an Iterator, then you can apply all the filter operations (or other things like map etc) in one pass:

    val list = (1 to 12).toList
    val doubleFiltered: List[Int] =
      list.iterator
        .filter(_ % 2 == 0)
        .filter(_ % 3 == 0)
        .toList
    println(doubleFiltered)
    

    When you convert the collection to an Iterator with .iterator, Scala will keep track of the operations to be performed (here, two filters), but will wait to perform them until the result is actually accessed (here, via the call to .toList).

    So I might rewrite your code like this:

    val list = (1 to 12).toList
    val evens = list.iterator.filter(_ % 2 == 0)
    
    val result = 
      if(someCondition)
        evens.filter(_ % 3 == 0)
      else
        evens.filter(_ % 5 == 0)
    
    result foreach println
    

    Depending on exactly what you want to do, you might want an Iterator, a Stream, or a View. They are all lazily computed (so the one-pass aspect will apply), but they differ on things like whether they can be iterated over multiple times (Stream and View) or whether they keep the computed value around for later access (Stream).

    To really see these different lazy behaviors, try running this bit of code and set to either toList, iterator, view, or toStream:

    val result =
      (1 to 12).
        .filter { e => println("filter 1: " + e); e % 2 == 0 }
        .filter { e => println("filter 2: " + e); e % 3 == 0 }
    result foreach println
    result foreach println
    

    Here's the behavior you will see:

    • List (or any other non-lazy collection): Each filter is requires a separate iteration through the collection. The resulting filtered collection is stored in memory so that each foreach can just display it.
    • Iterator: Both filters and the first foreach are done in a single iteration. The second foreach does nothing since the Iterator has been consumed. Results are not stored in memory.
    • View: Both foreach calls result in their own single-pass iteration over the collection to perform the filters. Results are not stored in memory.
    • Stream: Both filters and the first foreach are done in a single iteration. The resulting filtered collection is stored in memory so that each foreach can just display it.

提交回复
热议问题