第一章.java&golang的區別之:閉包

来源:https://www.cnblogs.com/mantu/archive/2019/02/15/10381316.html
-Advertisement-
Play Games

對於golang一直存有覬覦之心,但一直苦於沒有下定決心去學習研究,最近開始接觸golang。就我個人來說,學習golang的原動力是因為想要站在java語言之外來審視java和其它語言的區別,再就是想瞻仰一下如此NB的語言。年前就想在2019年做一件事情,希望能從各個細節處做一次java和gola ...


對於golang一直存有覬覦之心,但一直苦於沒有下定決心去學習研究,最近開始接觸golang。就我個人來說,學習golang的原動力是因為想要站在java語言之外來審視java和其它語言的區別,再就是想瞻仰一下如此NB的語言。年前就想在2019年做一件事情,希望能從各個細節處做一次java和golang的對比分析,不評判語言的優劣,只想用簡單的語言和可以隨時執行的代碼來表達出兩者的區別和底層涉及到的原理。今天是情人節,饅頭媽媽在加班,送給自己一件貼心的禮物,寫下第一篇對比文章:java&golang的區別之:閉包。 關於閉包到底是啥,建議參考知乎上的解釋:https://www.zhihu.com/question/51402215/answer/556617311

  • java8之前的閉包
在java8之前,java其實就已經對閉包有了一定層面的支持,實現的閉包方式主要是靠匿名類來實現的,下麵是java程式員經常寫的一段代碼:
 1 public class ClosureBeforeJava8 {
 2     int y = 1;
 3 
 4     public static void main(String[] args) {
 5         final int x = 0;
 6         ClosureBeforeJava8 closureBeforeJava8 = new ClosureBeforeJava8();
 7         Runnable run = closureBeforeJava8.getRunnable();
 8         new Thread(run).start();
 9     }
10 
11     public Runnable getRunnable() {
12         final int x = 0;
13         Runnable run = new Runnable() {
14             @Override
15             public void run() {
16  
17 System.out.println("local varable x is:" + x); 18 //System.out.println("member varable y is:" + this.y); //error 19 } 20 }; 21 return run; 22 } 23 }

上段代碼的輸出:local varable x is:0

在代碼的第13行到第20行,通過匿名類的方式實現了Runnable介面的run()方法,實現了一部分操作的集合(run方法),並將這些操作映射為java的對象,在java中就可以實現將函數以變數的方式進行傳遞了,如果僅僅是傳遞函數指針,那還不能算是閉包,我們再註意第17行代碼,在這段被封裝可以在不同的java對象間傳遞的代碼,引用了上層方法的局部變數,這個就有些閉包的意思在裡面了。但是第18行被註釋掉的代碼在匿名類的情況下卻無法編譯通過,也就是封裝的函數裡面,無法引用上層方法所在對象的成員變數。總結一下,java8之前的閉包特點如下:

1.可以實現封裝的函數在jvm里進行傳遞,可以在不同的對象里進行調用;

2.被封裝的函數,可以調用上層的方法里的局部變數,但是此局部變數必須為final,也就是不可以更改的(基礎類型不可以更改,引用類型不可以變更地址);

3.被封裝的函數,不可以調用上層方法所在對象的成員變數;
  • java8里對閉包的支持

java8里對於閉包的支持,其實也就是lamda表達式,我們再來看一下上段代碼在lamda表達式方式下的寫法:

 1 public class ClosureInJava8 {
 2     int y = 1;
 3 
 4     public static void main(String[] args) throws Exception{
 5         final int x = 0;
 6         ClosureInJava8 closureInJava8 = new ClosureInJava8();
 7         Runnable run = closureInJava8.getRunnable();
 8         Thread thread1 = new Thread(run);
 9         thread1.start();
10         thread1.join();
11         new Thread(run).start();
12     }
13 
14     public Runnable getRunnable() {
15         final int x = 0;
16         Runnable run = () -> {
17              
18 System.out.println("local varable x is:" + x); 19 System.out.println("member varable y is:" + this.y++); 20 }; 21 return run; 22 } 23 }

 上面對代碼輸出:

local varable x is:0
member varable y is:1
local varable x is:0
member varable y is:2

在代碼的第16行到第20行,通過lamda表達式的方式實現了函數的封裝(關於lamda表達式的用法,大家可以自行google)。通過代碼的輸出,大家可以發現,在lamda表達式的書寫方式下,封裝函數不但可以引用上層方法的effectively final類型(java8的特性之一,其實也是final類型)的局部變數,還可以引用上層方法所在對象的成員變數,並可以在其它線程和方法中對此成員變數進行修改。總結一下:java8對於閉包支持的特點如下:

1.通過lamda表達式的方式可以實現函數的封裝,並可以在jvm里進行傳遞;

2.lamda表達式,可以調用上層的方法里的局部變數,但是此局部變數必須為final或者是effectively final,也就是不可以更改的(基礎類型不可以更改,引用類型不可以變更地址);

3.lamda表達式,可以調用和修改上層方法所在對象的成員變數; 由於還沒時間分析jdk和hotspot的源碼,在此只能猜測推理,第2點和第3點的情況。關於第2點:上層方法的局部變數必須是final修飾的,網上的文章大部分都是說因為多線程併發的原因,無法在lamda表達式里進行修改上層方法的局部變數,這點上我是不同意這個觀點的。我認為主要原因是:java在定義局部變數時,對於基礎類型都是創建在stack frame上的,而一個方法執行完畢後,此方法所對應的stack frame也就沒有意義了,試想一下,lamda表達式所依賴的上層方法的局部變數的存儲區(stack frame)都消失了,我們還怎麼能夠修改這個變數,這是毫無意義的,在java里也很難實現這一點,除非像golang一下,在特定情況下,更改局部變數的存儲區域(在heap里存儲)。關於第3點:實現起來就比較容易,就是在lamda表達式的對象里,創建一個引用地址,地址指向原上層方法所在對象的堆存儲地址即可。
  • golang里對閉包的支持

golang里對於閉包的支持,理解起來就非常容易了,就是函數可以作為變數來傳遞使用,代碼如下:

 1 package main
 2 
 3 import "fmt"
 4 
 5 func main()  {
 6     ch := make(chan int ,1)
 7     ch2 := make(chan int ,1)
 8     fn := closureGet()
 9     go func() {
10         fn()
11         ch <-1
12     }()
13     go func() {
14         fn()
15         ch2 <-1
16     }()
17     <-ch
18     <-ch2
19 }
20 
21 func closureGet() func(){
22     x := 1
23     y := 2
24     fn := func(){
25         x = x +y
26         fmt.Printf("local varable x is:%d y is:%d \n", x, y)
27     }
28     return fn
29 }

代碼輸出如下:

local varable x is:3 y is:2
local varable x is:5 y is:2

代碼的第24行到27行,定義了一個方法fn,此方法可以使用上層方法的局部變數,總結一下:

1.golang的閉包在表達形式上,理解起來非常容易,就是函數可以作為變數,來直接傳遞;

2.golang的封裝函數可以沒有限制的使用上層函數里的局部變數,並且在不同的goroutine里修改的值,都會有所體現。

關於第2點,大家可以參考文章:https://studygolang.com/articles/11627  中關於golang閉包的講解部分。

  • 總結

golang的閉包從語言的簡潔性、理解的難易程度、支持的力度上來說,確實還是優於java的。本文作為java和golang對比分析的第一篇文章,由於調研分析的時間有限,難免有疏忽之處,歡迎各位指正。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 在 CSS3 之前,web 設計師必須使用已在用戶電腦上安裝好的字體。 通過 CSS3,web 設計師可以使用他們喜歡的任意字體。 當您找到或購買到希望使用的字體時,可將該字體文件存放到 web 伺服器上,它會在需要時被自動下載到用戶的電腦上。 您“自己的”的字體是在 CSS3 @font-fa ...
  • 簡介: 單例模式是一種簡單的設計模式,但是要在程式設計中使用好單例模式,卻需要註意幾個地方。 單例模式意味著在整個系統中,單例類只能有一個實例對象,且需要自行完成實例化,並始終對外提供同一個實例對象。 單例模式實現方式: 餓漢模式: 懶漢模式(單線程版): 上述懶漢模式有延遲載入的意思,但是沒有考慮 ...
  • 1、兩種創建方式 使用字面量創建時只會生成一個對象,而通過構造方法創建時會生成兩個對象(前面的str2和後面的new String對象) 2、常見的構造方法 3、其它常用方法 4、String、StringBuffer、StringBuilder 4.1、String是不可變的字元序列,在定義時長度 ...
  • 【New責任鏈&裝飾者】 感慨一下,本以為上下篇能一起發呢,結果要隔一定時間,傳統的責任鏈與裝飾者模式:https://www.cnblogs.com/SharePointApp/p/10340578.html 基本代碼 現在要做的就是責任鏈如果使用外置模式,能不能像裝飾者一樣幾個處理類聯合處理?答 ...
  • peewee 是一個輕量級的 Python ORM 框架,個人用下來感覺還好,簡單易上手,對於小項目能滿足大部分需求。peewee 的官方文檔沒有中文版的,網上的文章都是快速入門,抄了文檔中幾個例子,沒有詳細介紹的。本文略長,詳細介紹了增刪改查操作,並且詳解介紹了使用過程中遇到的各種問題。 ...
  • 爬前叨叨 已經編寫了33篇爬蟲文章了,如果你按著一個個的實現,你的爬蟲技術已經入門,從今天開始慢慢的就要寫一些有分析價值的數據了,今天我選了一個《掘金網》,我們去爬取一下他的 全站用戶 數據。 爬取思路 獲取全站用戶,理論來說從1個用戶作為切入點就可以,我們需要爬取用戶的關註列表,從關註列表不斷的疊 ...
  • 題意 "題目鏈接" Sol 這個東西的學名應該叫“閔可夫斯基和”。就是合併兩個凸包 首先我們先分別求出給出的兩個多邊形的凸包。合併的時候直接拿個雙指針掃一下,每次選最凸的點就行了。 複雜度$O(nlogn + n)$ cpp include define LL long long // define ...
  • 故事的開篇 把時鐘撥回到9年前,那是2010年的10月份,李大胖入職了新公司,在公司本部呆了一個多月,就被髮往客戶單位進行駐場開發。 公司給客戶做的系統,需要不斷的二次開發和維護,所以直接去客戶單位比較利於工作的進行。 一個周五的下午,經理把我叫到辦公室,說到:“下周你就去客戶現場吧,我們有些同事在 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...