kotlin之閉包案例詳解
閉包,函數(shù)式編程福音
先了解函數(shù)式編程(Functional Programming)
概念:它屬于“結(jié)構(gòu)化編程”的一種,主要思想是把運(yùn)算過程盡量寫成一系列嵌套的函數(shù)調(diào)用。函數(shù)式編程語言最重要的基礎(chǔ)是λ運(yùn)算(Lambda表達(dá)式),λ運(yùn)算的函數(shù)可以接受函數(shù)當(dāng)做參數(shù)或返回值。
對(duì)比函數(shù)式編程與面向?qū)ο缶幊?/strong>
面向?qū)ο缶幊蹋∣bject-oriented programming,縮寫OOP)
面向?qū)ο缶幊淌且环N具有對(duì)象概念的程序編程范型,它可能包含數(shù)據(jù)、屬性、方法。它將對(duì)象作為程序的基本單元,將方法和數(shù)據(jù)封裝其中,以提高軟件的重用性、靈活性和擴(kuò)展性。對(duì)象里的程序可以訪問及經(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)的面向過程編程將程序看作一系列函數(shù)的集合這種無系統(tǒng)化和結(jié)構(gòu)化的模式,面向?qū)ο缶幊虒⒁幌盗嘘P(guān)聯(lián)性的數(shù)據(jù)、方法結(jié)構(gòu)化,封裝成類,通過類的對(duì)象進(jìn)行方法、屬性調(diào)用的方式,可以讓編程者更加便于分析、設(shè)計(jì)和理解。
2.程序的靈活性和可維護(hù)性
面向?qū)ο缶幊逃捎诩?、封裝、多態(tài)的特性,可以更好的設(shè)計(jì)出高內(nèi)聚、低耦合的系統(tǒng) 結(jié)構(gòu),使得系統(tǒng)更靈活、更容易擴(kuò)展,開發(fā)及維護(hù)成本更低。
面向?qū)ο缶幊痰娜秉c(diǎn)
- 運(yùn)行效率較低
面向?qū)ο箅m然開發(fā)效率高但是代碼運(yùn)行效率比起面向過程要低很多,這也限制了面向?qū)ο蟮氖褂脠?chǎng)景不能包括那些對(duì)性能要求很苛刻的地方 - 多線程數(shù)據(jù)不安全
面向?qū)ο缶幊桃詳?shù)據(jù)為核心,所以在多線程并發(fā)編程中,多個(gè)線程同時(shí)操作數(shù)據(jù)的時(shí)候可能會(huì)導(dǎo)致數(shù)據(jù)修改的不確定性。
函數(shù)式編程優(yōu)點(diǎn)(可以說就是為了解決面向?qū)ο蟮娜秉c(diǎn)問題而設(shè)計(jì)的)
- 線程安全
在函數(shù)式編程中,數(shù)據(jù)全部都是不可變的,所以沒有并發(fā)編程的問題,是多線程安全的,可以有效降低程序運(yùn)行中所產(chǎn)生的副作用。對(duì)于快速迭代的項(xiàng)目來說,函數(shù)式編程可以實(shí)現(xiàn)函數(shù)與函數(shù)之間的熱切換而不用擔(dān)心數(shù)據(jù)的問題,因?yàn)樗且院瘮?shù)作為最小單位的,只要函數(shù)與函數(shù)的關(guān)系正確即可保證結(jié)果的正確性。 - 代碼可讀性高
函數(shù)式編程的表達(dá)方式更加符合人類日常生活中的語法,代碼可讀性更強(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)代的汗水編程語言使用了很多技巧,比如惰性計(jì)算等優(yōu)化運(yùn)行速度,但始終無法與面向?qū)ο笙啾?,?dāng)然比面向過程的程序就更慢了
了解完函數(shù)式編程,再回歸今天的主題——閉包
什么是閉包?
我們都知道,程序的變量分為全局變量和局部變量,全局變量,顧名思義,其作用域是當(dāng)前文件甚至文件外的所有地方;而局部變量,我們只能再其有限的作用域里獲取。
那么,如何在外部調(diào)用局部變量呢?答案就是——閉包,與此給閉包下個(gè)定義:閉包就是能夠讀取其他函數(shù)內(nèi)部變量的函數(shù)
- 它是運(yùn)行的環(huán)境
- 它持有函數(shù)的運(yùn)行狀態(tài)
- 它的內(nèi)部可以定義函數(shù)
- 它的內(nè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類型的函數(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ī)硪黄赟pring@Autowired注解與自動(dòng)裝配詳談。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-10-10MyBatis Plus構(gòu)建一個(gè)簡(jiǎn)單的項(xiàng)目的實(shí)現(xiàn)
這篇文章主要介紹了MyBatis Plus構(gòu)建一個(gè)簡(jiǎn)單的項(xiàng)目的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-11-11Java設(shè)計(jì)模式中的單一責(zé)任原則詳解
這篇文章主要介紹了Java設(shè)計(jì)模式中的單一責(zé)任原則詳解,應(yīng)該有且僅有一個(gè)原因引起類的變更,即單一指責(zé)原則要求一個(gè)借口或類只有一個(gè)原因引起變化,也就是一個(gè)接口或類只有一個(gè)職責(zé),它就負(fù)責(zé)一件事情,需要的朋友可以參考下2023-11-11springboot集成@DS注解實(shí)現(xiàn)數(shù)據(jù)源切換的方法示例
本文主要介紹了springboot集成@DS注解實(shí)現(xiàn)數(shù)據(jù)源切換的方法示例,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-03-03java創(chuàng)建子類對(duì)象設(shè)置并調(diào)用父類的變量操作
這篇文章主要介紹了java創(chuàng)建子類對(duì)象設(shè)置并調(diào)用父類的變量操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2021-01-01java.net.ConnectException: Connection refused問題解決辦法
這篇文章主要介紹了java.net.ConnectException: Connection refused問題解決辦法的相關(guān)資料,需要的朋友可以參考下2016-12-12基于SpringBoot開機(jī)啟動(dòng)與@Order注解
這篇文章主要介紹了SpringBoot開機(jī)啟動(dòng)與@Order注解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09