kotlin之閉包案例詳解
閉包,函數(shù)式編程福音
先了解函數(shù)式編程(Functional Programming)
概念:它屬于“結(jié)構(gòu)化編程”的一種,主要思想是把運(yùn)算過(guò)程盡量寫(xiě)成一系列嵌套的函數(shù)調(diào)用。函數(shù)式編程語(yǔ)言最重要的基礎(chǔ)是λ運(yùn)算(Lambda表達(dá)式),λ運(yùn)算的函數(shù)可以接受函數(shù)當(dāng)做參數(shù)或返回值。
對(duì)比函數(shù)式編程與面向?qū)ο缶幊?/strong>
面向?qū)ο缶幊蹋∣bject-oriented programming,縮寫(xiě)OOP)
面向?qū)ο缶幊淌且环N具有對(duì)象概念的程序編程范型,它可能包含數(shù)據(jù)、屬性、方法。它將對(duì)象作為程序的基本單元,將方法和數(shù)據(jù)封裝其中,以提高軟件的重用性、靈活性和擴(kuò)展性。對(duì)象里的程序可以訪(fǎng)問(wèn)及經(jīng)常修改對(duì)象相關(guān)聯(lián)的數(shù)據(jù)。在面向?qū)ο缶幊汤?,?jì)算機(jī)程序會(huì)被設(shè)計(jì)成彼此相關(guān)的對(duì)象。
面向?qū)ο缶幊痰膬?yōu)點(diǎn)
1.程序的結(jié)構(gòu)化
面向?qū)ο蟪绦蛟O(shè)計(jì)可以看作一種在程序中包含各種獨(dú)立而又互相調(diào)用的對(duì)象的思想,相比傳統(tǒng)的面向過(guò)程編程將程序看作一系列函數(shù)的集合這種無(wú)系統(tǒng)化和結(jié)構(gòu)化的模式,面向?qū)ο缶幊虒⒁幌盗嘘P(guān)聯(lián)性的數(shù)據(jù)、方法結(jié)構(gòu)化,封裝成類(lèi),通過(guò)類(lèi)的對(duì)象進(jìn)行方法、屬性調(diào)用的方式,可以讓編程者更加便于分析、設(shè)計(jì)和理解。
2.程序的靈活性和可維護(hù)性
面向?qū)ο缶幊逃捎诩伞⒎庋b、多態(tài)的特性,可以更好的設(shè)計(jì)出高內(nèi)聚、低耦合的系統(tǒng) 結(jié)構(gòu),使得系統(tǒng)更靈活、更容易擴(kuò)展,開(kāi)發(fā)及維護(hù)成本更低。
面向?qū)ο缶幊痰娜秉c(diǎn)
- 運(yùn)行效率較低
面向?qū)ο箅m然開(kāi)發(fā)效率高但是代碼運(yùn)行效率比起面向過(guò)程要低很多,這也限制了面向?qū)ο蟮氖褂脠?chǎng)景不能包括那些對(duì)性能要求很苛刻的地方 - 多線(xiàn)程數(shù)據(jù)不安全
面向?qū)ο缶幊桃詳?shù)據(jù)為核心,所以在多線(xiàn)程并發(fā)編程中,多個(gè)線(xiàn)程同時(shí)操作數(shù)據(jù)的時(shí)候可能會(huì)導(dǎo)致數(shù)據(jù)修改的不確定性。
函數(shù)式編程優(yōu)點(diǎn)(可以說(shuō)就是為了解決面向?qū)ο蟮娜秉c(diǎn)問(wèn)題而設(shè)計(jì)的)
- 線(xiàn)程安全
在函數(shù)式編程中,數(shù)據(jù)全部都是不可變的,所以沒(méi)有并發(fā)編程的問(wèn)題,是多線(xiàn)程安全的,可以有效降低程序運(yùn)行中所產(chǎn)生的副作用。對(duì)于快速迭代的項(xiàng)目來(lái)說(shuō),函數(shù)式編程可以實(shí)現(xiàn)函數(shù)與函數(shù)之間的熱切換而不用擔(dān)心數(shù)據(jù)的問(wèn)題,因?yàn)樗且院瘮?shù)作為最小單位的,只要函數(shù)與函數(shù)的關(guān)系正確即可保證結(jié)果的正確性。 - 代碼可讀性高
函數(shù)式編程的表達(dá)方式更加符合人類(lèi)日常生活中的語(yǔ)法,代碼可讀性更強(qiáng)。實(shí)現(xiàn)同樣的功能函數(shù)式編程所需要的代碼比面向?qū)ο缶幊桃俸芏?,代碼更加簡(jiǎn)潔明晰。
函數(shù)式編程的缺點(diǎn)
運(yùn)行速度更慢
由于所有的數(shù)據(jù)都是不可變的,所有的變量在程序運(yùn)行期間都是一直存在的,非常占用運(yùn)行資源。同時(shí)由于函數(shù)式的先天性設(shè)計(jì)導(dǎo)致性能一直不夠。雖然現(xiàn)代的汗水編程語(yǔ)言使用了很多技巧,比如惰性計(jì)算等優(yōu)化運(yùn)行速度,但始終無(wú)法與面向?qū)ο笙啾?,?dāng)然比面向過(guò)程的程序就更慢了
了解完函數(shù)式編程,再回歸今天的主題——閉包
什么是閉包?
我們都知道,程序的變量分為全局變量和局部變量,全局變量,顧名思義,其作用域是當(dāng)前文件甚至文件外的所有地方;而局部變量,我們只能再其有限的作用域里獲取。
那么,如何在外部調(diào)用局部變量呢?答案就是——閉包,與此給閉包下個(gè)定義:閉包就是能夠讀取其他函數(shù)內(nèi)部變量的函數(shù)
- 它是運(yùn)行的環(huán)境
- 它持有函數(shù)的運(yùn)行狀態(tài)
- 它的內(nèi)部可以定義函數(shù)
- 它的內(nèi)部也可以定義類(lèi)
首先看個(gè)簡(jiǎn)單的例子
//這是一個(gè)返回值為一個(gè)函數(shù)的高階函數(shù)
fun makeFun():()->Unit{
var conut = 0
return fun(){ //返回一個(gè)匿名函數(shù),這個(gè)函數(shù)持有count的狀態(tài)
println(++conut)
}
}
fun main() {
val makeFun = makeFun() //函數(shù)調(diào)用,返回一個(gè)函數(shù)
makeFun() //調(diào)用這個(gè)返回的函數(shù),此時(shí)makeFun持有makeFun()內(nèi)部變量的狀態(tài)
makeFun()
makeFun()
}
運(yùn)行結(jié)果:

在比如一個(gè)稍微復(fù)雜一點(diǎn)的例子,實(shí)現(xiàn)斐波那契數(shù)列
//斐波那契數(shù)列
fun fibonacci():()->Long{
var first = 0L
var second = 1L
return fun():Long{ //返回返回值為L(zhǎng)ong類(lèi)型的函數(shù)
val result = second
second += first
first = second - first
return result
}
}
fun main() {
val fibo = fibonacci() //此時(shí),這個(gè)返回的函數(shù)fibo持有fibonnacci()函數(shù)內(nèi)部變量的狀態(tài)
println(fibo())
println(fibo())
println(fibo())
println(fibo())
println(fibo())
}
測(cè)試運(yùn)行:

使用迭代器實(shí)現(xiàn)斐波那契數(shù)列
//使用迭代器實(shí)現(xiàn)斐波那契數(shù)列(這里就不是返回一個(gè)函數(shù)而是一個(gè)對(duì)象了)
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)
}
}
運(yùn)行結(jié)果:

到此這篇關(guān)于kotlin之閉包案例詳解的文章就介紹到這了,更多相關(guān)kotlin之閉包內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
基于Spring@Autowired注解與自動(dòng)裝配詳談
下面小編就為大家?guī)?lái)一篇基于Spring@Autowired注解與自動(dòng)裝配詳談。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-10-10
MyBatis Plus構(gòu)建一個(gè)簡(jiǎn)單的項(xiàng)目的實(shí)現(xiàn)
這篇文章主要介紹了MyBatis Plus構(gòu)建一個(gè)簡(jiǎn)單的項(xiàng)目的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-11-11
Java設(shè)計(jì)模式中的單一責(zé)任原則詳解
這篇文章主要介紹了Java設(shè)計(jì)模式中的單一責(zé)任原則詳解,應(yīng)該有且僅有一個(gè)原因引起類(lèi)的變更,即單一指責(zé)原則要求一個(gè)借口或類(lèi)只有一個(gè)原因引起變化,也就是一個(gè)接口或類(lèi)只有一個(gè)職責(zé),它就負(fù)責(zé)一件事情,需要的朋友可以參考下2023-11-11
springboot集成@DS注解實(shí)現(xiàn)數(shù)據(jù)源切換的方法示例
本文主要介紹了springboot集成@DS注解實(shí)現(xiàn)數(shù)據(jù)源切換的方法示例,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03
又一波Java專(zhuān)業(yè)人士必備書(shū)籍來(lái)襲
又一波Java專(zhuān)業(yè)人士必備書(shū)籍來(lái)襲,這篇文章主要向大家推薦了Java專(zhuān)業(yè)人士必讀的書(shū),感興趣的小伙伴們不要錯(cuò)過(guò)2016-09-09
java創(chuàng)建子類(lèi)對(duì)象設(shè)置并調(diào)用父類(lèi)的變量操作
這篇文章主要介紹了java創(chuàng)建子類(lèi)對(duì)象設(shè)置并調(diào)用父類(lèi)的變量操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-01-01
java.net.ConnectException: Connection refused問(wèn)題解決辦法
這篇文章主要介紹了java.net.ConnectException: Connection refused問(wèn)題解決辦法的相關(guān)資料,需要的朋友可以參考下2016-12-12
基于SpringBoot開(kāi)機(jī)啟動(dòng)與@Order注解
這篇文章主要介紹了SpringBoot開(kāi)機(jī)啟動(dòng)與@Order注解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09

