当前位置: 首页 > news >正文

Swift Collection 中的 lazy 作用

惰性求值

惰性求值常见于函数式编程中,也有人把惰性求值翻译成延迟求值(Lazy Evaluation)。它的目的是要最小化计算机要做的工作。在使用惰性求值的时候,表达式不在它被绑定到变量之后就立即求值,而是在该值被取用的时候求值。 惰性求值的优点很明显:

  • 计算只有在真正需要的时候才会执行 这个特性就给代码优化提供了更多的可能性。有些运算可能在某个条件分支下才会被用到。某些计算可能和后面的计算相抵消,可以优化运算的方式。写一段伪代码大概如下:
let a = 100
let b = 1000
let c = a/b
print(c*b) 
复制代码

最后求值的时候发现表达式是a/b*b,所以最后输出的时候的值就为a,省掉了除b,乘b的运算。

  • 无穷的数据结构 因为在没被使用的时候只是一个表达式,在程序中真正使用某几项具体数据的时候才会去计算,所以惰性求值可以声明一个无穷的序列。 下面的代码先定义了一个从 1 开始的奇数序列:
var nums = sequence(first: 1, next: { num in
    return num + 2
})

// 这个时候才进行真正的计算
for n in nums.prefix(5) {
    print(n)
}
复制代码

Swift 中的 lazy

Swift 的编程范式虽然有函数式编程,但是 swift 和纯函数式编程的语言如 haskell 还是有一些差距。Swift 中集合操作默认的求值策略是 eager,就是及早求值。当表达式赋值给一个变量时计算就进行了。 为了支持惰性求值,在 collection 通过 lazy 属性可以获取到一个惰性的集合。

// a huge collection
let giant = 0..<Int.max
// lazily map it: no work is done yet
let mapped = giant.lazy.map { $0 * 2 }
// sum the first few elements
let sum = mapped.prefix(10).reduce(0, +)
// sum == 90
复制代码

上面的代码中,mapped 赋值时计算并没有发生,因为 lazy 返回的 LazyCollecion 上的 map 操作是惰性的。

取舍

惰性求值的缺点就和异步一样,写代码的时候不能显示的看出执行顺序。 在有些场景是需要及时求值的。比如下面的代码在惰性求值的体系下第一行不需要在第二行前执行:

System.out.println("Please enter your name: ");
System.in.readLine();
复制代码

因此一个编程语言也不能只支持惰性求值。只是在 swift 中,苹果认为大部分的时候及早求值就能满足场景,需要通过显式的 lazy 获取惰性集合。当然也可能在 swift 前期特性中,函数式的一些特性不是最核心的需求,期待后期的 swift 特性吧。


引用

  • 惰性求值
  • Conditional Conformance in the Standard Library
  • 《傻瓜函数式编程》

欢迎关注我的微博:@没故事的卓同学

掘金博客地址:juejin.im/user/5624c8…

如果想与我有更密切的交流也可以加入我的小密圈:程序员生存指南

相关文章:

  • Activiti用户任务分配
  • 2 使用unitest 模块扩展功能测试
  • 免费学习coursera的课程的操作办法
  • SylixOS Lite版本中断优先级设置
  • 小程序的iphoneX判断
  • 还在担心机器人?人工智能目前水平还不如初中生
  • linux多网卡绑定聚合-Bond详细完整版
  • HDU_1237_简单计算器
  • Docker基础技术:DeviceMapper
  • Docker社区版中Kubernetes开发
  • 云栖科技评论第38期:亚马逊推出时尚智能助手Echo Look
  • hadoop生态系统学习之路(十)MR将结果输出到hbase
  • Python入门篇(二)
  • 我为什么突然想转管理?
  • Eclipse Open J9:Eclipse OMR项目提供的开源JVM
  • [分享]iOS开发 - 实现UITableView Plain SectionView和table不停留一起滑动
  • 【技术性】Search知识
  • github从入门到放弃(1)
  • JavaScript/HTML5图表开发工具JavaScript Charts v3.19.6发布【附下载】
  • JSONP原理
  • JS专题之继承
  • LeetCode29.两数相除 JavaScript
  • MQ框架的比较
  • OpenStack安装流程(juno版)- 添加网络服务(neutron)- controller节点
  • PaddlePaddle-GitHub的正确打开姿势
  • PHP的类修饰符与访问修饰符
  • Redux系列x:源码分析
  • Spring Cloud中负载均衡器概览
  • STAR法则
  • 服务器之间,相同帐号,实现免密钥登录
  • 精益 React 学习指南 (Lean React)- 1.5 React 与 DOM
  • 聊一聊前端的监控
  • 前端面试之CSS3新特性
  • 使用agvtool更改app version/build
  • 想晋级高级工程师只知道表面是不够的!Git内部原理介绍
  • 鱼骨图 - 如何绘制?
  • 源码安装memcached和php memcache扩展
  • linux 淘宝开源监控工具tsar
  • 容器镜像
  • 昨天1024程序员节,我故意写了个死循环~
  • ​渐进式Web应用PWA的未来
  • (Matalb时序预测)PSO-BP粒子群算法优化BP神经网络的多维时序回归预测
  • (pt可视化)利用torch的make_grid进行张量可视化
  • (TOJ2804)Even? Odd?
  • (免费分享)基于springboot,vue疗养中心管理系统
  • (一)Java算法:二分查找
  • (一)基于IDEA的JAVA基础1
  • (状压dp)uva 10817 Headmaster's Headache
  • ******之网络***——物理***
  • **Java有哪些悲观锁的实现_乐观锁、悲观锁、Redis分布式锁和Zookeeper分布式锁的实现以及流程原理...
  • .class文件转换.java_从一个class文件深入理解Java字节码结构
  • .helper勒索病毒的最新威胁:如何恢复您的数据?
  • .Mobi域名介绍
  • .Net 6.0 处理跨域的方式
  • .net core 微服务_.NET Core 3.0中用 Code-First 方式创建 gRPC 服务与客户端