时间:2022-08-10 09:39:15 | 栏目:JAVA代码 | 点击:次
闭包,函数式编程福音
先了解函数式编程(Functional Programming)
概念:它属于“结构化编程”的一种,主要思想是把运算过程尽量写成一系列嵌套的函数调用。函数式编程语言最重要的基础是λ运算(Lambda表达式),λ运算的函数可以接受函数当做参数或返回值。
对比函数式编程与面向对象编程
面向对象编程(Object-oriented programming,缩写OOP)
面向对象编程是一种具有对象概念的程序编程范型,它可能包含数据、属性、方法。它将对象作为程序的基本单元,将方法和数据封装其中,以提高软件的重用性、灵活性和扩展性。对象里的程序可以访问及经常修改对象相关联的数据。在面向对象编程里,计算机程序会被设计成彼此相关的对象。
面向对象程序设计可以看作一种在程序中包含各种独立而又互相调用的对象的思想,相比传统的面向过程编程将程序看作一系列函数的集合这种无系统化和结构化的模式,面向对象编程将一系列关联性的数据、方法结构化,封装成类,通过类的对象进行方法、属性调用的方式,可以让编程者更加便于分析、设计和理解。
面向对象编程由于集成、封装、多态的特性,可以更好的设计出高内聚、低耦合的系统 结构,使得系统更灵活、更容易扩展,开发及维护成本更低。
运行速度更慢
由于所有的数据都是不可变的,所有的变量在程序运行期间都是一直存在的,非常占用运行资源。同时由于函数式的先天性设计导致性能一直不够。虽然现代的汗水编程语言使用了很多技巧,比如惰性计算等优化运行速度,但始终无法与面向对象相比,当然比面向过程的程序就更慢了
了解完函数式编程,再回归今天的主题――闭包
我们都知道,程序的变量分为全局变量和局部变量,全局变量,顾名思义,其作用域是当前文件甚至文件外的所有地方;而局部变量,我们只能再其有限的作用域里获取。
那么,如何在外部调用局部变量呢?答案就是――闭包,与此给闭包下个定义:闭包就是能够读取其他函数内部变量的函数
首先看个简单的例子
//这是一个返回值为一个函数的高阶函数 fun makeFun():()->Unit{ var conut = 0 return fun(){ //返回一个匿名函数,这个函数持有count的状态 println(++conut) } } fun main() { val makeFun = makeFun() //函数调用,返回一个函数 makeFun() //调用这个返回的函数,此时makeFun持有makeFun()内部变量的状态 makeFun() makeFun() }
运行结果:
在比如一个稍微复杂一点的例子,实现斐波那契数列
//斐波那契数列 fun fibonacci():()->Long{ var first = 0L var second = 1L return fun():Long{ //返回返回值为Long类型的函数 val result = second second += first first = second - first return result } } fun main() { val fibo = fibonacci() //此时,这个返回的函数fibo持有fibonnacci()函数内部变量的状态 println(fibo()) println(fibo()) println(fibo()) println(fibo()) println(fibo()) }
测试运行:
使用迭代器实现斐波那契数列
//使用迭代器实现斐波那契数列(这里就不是返回一个函数而是一个对象了) fun fibonacci2():Iterable<Long>{ var first = 0L var second = 1L return Iterable { object :LongIterator(){ override fun hasNext() = true override fun nextLong(): Long { val result = second second += first first = second - first return result } } } } fun main() { val fibo2 = fibonacci2() for (i in fibo2){ if (i>60) break println(i) } }
运行结果: