加入收藏 | 设为首页 | 会员中心 | 我要投稿 李大同 (https://www.lidatong.com.cn/)- 科技、建站、经验、云计算、5G、大数据,站长网!
当前位置: 首页 > 综合聚焦 > 服务器 > 安全 > 正文

迭代器上的Scala贴图不会产生副作用

发布时间:2020-12-16 19:09:52 所属栏目:安全 来源:网络整理
导读:为什么会这样, scala List(1,2,3,4).iterator.map((x: Int) = println(x)) 不打印出来 1234 而 List(1,4).map((x: Int) = println(x))List(1,4).foreach((x: Int) = println(x))List(1,4).iterator.foreach((x: Int) = println(x)) 都这样做? 换句话说,为什
为什么会这样,

scala> List(1,2,3,4).iterator.map((x: Int) => println(x))

不打印出来

1
2
3
4

List(1,4).map((x: Int) => println(x))
List(1,4).foreach((x: Int) => println(x))
List(1,4).iterator.foreach((x: Int) => println(x))

都这样做?

换句话说,为什么迭代器上的地图将类型T映射到单位并且副作用无法显示这些副作用?

编辑:

另外,为什么如果迭代器是惰性的,下面的lazyMap调用实际上会从头到尾计算新的迭代器(提供完整的新迭代器)?

def lazyMap[T,U](coll: Iterable[T],f: T => U) = new Iterable[U] {
  def iterator = coll.iterator map f
}

scala> lazyMap(List(1,4),(x: Int) => x + 1)
res4: java.lang.Object with Iterable[Int] = (2,4,5)

解决方法

迭代器上的原因映射是懒惰的,你需要一些严格性:

scala> List(1,4).iterator.map((x: Int) => println(x))
res0: Iterator[Unit] = non-empty iterator

// nothing actually happened yet,just remember to do this printing things

scala> res0.toList
1
2
3
4
res1: List[Unit] = List((),(),())

当你在迭代器上做foreach时很明显你正在做副作用,所以懒惰将是不受欢迎的.关于地图,我不会这么说.

UPD

至于你的编辑:这种行为的原因是,对于语句结果存在toString的隐式调用,而这反过来又会严格破坏迭代器 – 试试你自己的代码:

scala> { lazyMap(List(1,{(x: Int) => println(x); x + 1}); 1 }

你会看到函数f永远不会被调用

(编辑:李大同)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    推荐文章
      热点阅读