go語言中的defer Go語言中的Handle

Go語言文件操作

本文主要介紹了Go語言中文件讀寫的相關操作。

為廊坊等地區用戶提供了全套網頁設計制作服務,及廊坊網站建設行業解決方案。主營業務為成都網站建設、成都做網站、廊坊網站設計,以傳統方式定制建設網站,并提供域名空間備案等一條龍服務,秉承以專業、用心的態度為用戶提供真誠的服務。我們深信只要達到每一位用戶的要求,就會得到認可,從而選擇與我們長期合作。這樣,我們也可以走得更遠!

文件是什么?

計算機中的文件是存儲在外部介質(通常是磁盤)上的數據集合,文件分為文本文件和二進制文件。

os.Open() 函數能夠打開一個文件,返回一個 *File 和一個 err 。對得到的文件實例調用 close() 方法能夠關閉文件。

為了防止文件忘記關閉,我們通常使用defer注冊文件關閉語句。

Read方法定義如下:

它接收一個字節切片,返回讀取的字節數和可能的具體錯誤,讀到文件末尾時會返回 0 和 io.EOF 。 舉個例子:

使用for循環讀取文件中的所有數據。

bufio是在file的基礎上封裝了一層API,支持更多的功能。

io/ioutil 包的 ReadFile 方法能夠讀取完整的文件,只需要將文件名作為參數傳入。

os.OpenFile() 函數能夠以指定模式打開文件,從而實現文件寫入相關功能。

其中:

name :要打開的文件名 flag :打開文件的模式。 模式有以下幾種:

perm :文件權限,一個八進制數。r(讀)04,w(寫)02,x(執行)01。

golang怎么捕獲c代碼的異常

1、Painc用法是:用于拋出錯誤。Recover()用法是:將Recover()寫在defer中,并且在可能發生

panic的地方之前,先調用此defer的東西(讓系統方法域結束時,有代碼要執行。)當程序遇到panic的時候(當然,也可以正常的調用出現的異常

情況),系統將跳過后面的代碼,進入defer,如果defer函數中recover(),則返回捕獲到的panic的值。

2、代碼:

package main

import "fmt"

func main() {

fmt.Printf("hello world my name is %s, I'm %d\r\n", "songxingzhu", 26)

defer func() {

if err := recover(); err != nil {

fmt.Println("出了錯:", err)

}

}()

myPainc()

fmt.Printf("這里應該執行不到!")

}

func myPainc() {

var x = 30

var y = 0

//panic("我就是一個大錯誤!")

var c = x / y

fmt.Println(c)

}

 

3、執行結果:

Atom Runner: main.go

hello world my name is songxingzhu, I'm 26

出了錯: runtime error: integer divide by zeroExited with code=0 in 1.667 seconds

GO語言學習系列八——GO函數(func)的聲明與使用

GO是編譯性語言,所以函數的順序是無關緊要的,為了方便閱讀,建議入口函數 main 寫在最前面,其余函數按照功能需要進行排列

GO的函數 不支持嵌套,重載和默認參數

GO的函數 支持 無需聲明變量,可變長度,多返回值,匿名,閉包等

GO的函數用 func 來聲明,且左大括號 { 不能另起一行

一個簡單的示例:

輸出為:

參數:可以傳0個或多個值來供自己用

返回:通過用 return 來進行返回

輸出為:

上面就是一個典型的多參數傳遞與多返回值

對例子的說明:

按值傳遞:是對某個變量進行復制,不能更改原變量的值

引用傳遞:相當于按指針傳遞,可以同時改變原來的值,并且消耗的內存會更少,只有4或8個字節的消耗

在上例中,返回值 (d int, e int, f int) { 是進行了命名,如果不想命名可以寫成 (int,int,int){ ,返回的結果都是一樣的,但要注意:

當返回了多個值,我們某些變量不想要,或實際用不到,我們可以使用 _ 來補位,例如上例的返回我們可以寫成 d,_,f := test(a,b,c) ,我們不想要中間的返回值,可以以這種形式來舍棄掉

在參數后面以 變量 ... type 這種形式的,我們就要以判斷出這是一個可變長度的參數

輸出為:

在上例中, strs ...string 中, strs 的實際值是b,c,d,e,這就是一個最簡單的傳遞可變長度的參數的例子,更多一些演變的形式,都非常類似

在GO中 defer 關鍵字非常重要,相當于面相對像中的析構函數,也就是在某個函數執行完成后,GO會自動這個;

如果在多層循環中函數里,都定義了 defer ,那么它的執行順序是先進后出;

當某個函數出現嚴重錯誤時, defer 也會被調用

輸出為

這是一個最簡單的測試了,當然還有更復雜的調用,比如調試程序時,判斷是哪個函數出了問題,完全可以根據 defer 打印出來的內容來進行判斷,非常快速,這種留給你們去實現

一個函數在函數體內自己調用自己我們稱之為遞歸函數,在做遞歸調用時,經常會將內存給占滿,這是非常要注意的,常用的比如,快速排序就是用的遞歸調用

本篇重點介紹了GO函數(func)的聲明與使用,下一篇將介紹GO的結構 struct

GoLang中defer的作用域不是塊,而是函數,是出于怎樣的考慮

舉個例子,如果我們的代碼邏輯是下面這樣的:

打開數據庫連接

defer 關閉連接

defer 刪除數據

因為一般defer定義是和打開連接并列的,打開文件,打開連接之后就定義了defer, 如果這之后你的defer是基于這個連接做的事情,那么如果先進先執行的話就會錯誤了。這就是當初Go設計defer的時候考慮的問題。

這里順帶提醒一下defer是存在一些小坑的,就是defer里面的變量是申明的時候就copy的,不會隨著后面的函數邏輯改變而改變,除非你用指針類型。

package main

import "fmt"

func main() {

var whatever [5]struct{}

for i := range whatever {

fmt.Println(i)

}

for i := range whatever {

defer func() { fmt.Println(i) }()

}

for i := range whatever {

defer func(n int) { fmt.Println(n) }(i)

}

}

從PHP 到Golang 的筆記 ( 轉 )

———文章來源 YamiOdymel/PHP-to-Golang

PHP和模塊之間的關系令人感到煩躁,假設你要讀取 yaml 檔案,你需要有一個 yaml 的模塊,為此,你還需要將其編譯然后將編譯后的模塊擺放至指定位置,之后換了一臺伺服器你還要重新編譯,這點到現在還是沒有改善;順帶一提之后出了PHP 7效能確實提升了許多(比Python 3快了些),但PHP仍令我感到臃腫,我覺得是時候

(轉行)了。

PHP 和Golang 的效能我想毋庸置疑是后者比較快(而且是以倍數來算),也許有的人會認為兩種不應該被放在一起比較,但Golang 本身就是偏向Web 開發的,所以這也是為什么我考慮轉用Golang 的原因,起初我的考慮有幾個:Node.js 和Rust 還有最終被選定的Golang;先談談Node.js 吧。

Node.js的效能可以說是快上PHP 3.5倍至6倍左右 ,而且撰寫的語言還是JavaScript,蒸蚌,如此一來就不需要學習新語言了!搭配Babel更可以說是萬能,不過那跟「跳跳虎」一樣的Async邏輯還有那恐怖的Callback Hell,有人認為前者是種優點,這點我不否認,但是對學習PHP的我來說太過于"Mind Fuck",至于后者的Callback Hell雖然有Promise,但是那又是另一個「Then Hell」的故事了。相較于Golang之下,Node.js似乎就沒有那么吸引我了。你確實可以用Node.js寫出很多東西,不過那V8引擎的效能仍然有限,而且要學習新的事物,不就應該是「全新」的嗎;)?

題外話: 為什么Node.js不適合大型和商業專案?

在拋棄改用Node.js 之后我曾經花了一天的時間嘗試Rust 和Iron 框架,嗯??Rust 太強大了,強大到讓我覺得Rust 不應該用在這里,這想法也許很蠢,但Rust 讓我覺得適合更應該拿來用在系統或者是部分底層的地方,而不應該是網路服務。

Golang是我最終的選擇,主要在于我花了一天的時間來研究的時候意外地發現Golang夭壽簡潔( 關鍵字只有25個 ),相較之下Rust太過于「強大」令我怯步;而且Golang帶有許多工具,例如 go fmt 會自動幫你整理程式碼、 go doc 會自動幫你生產文件、 go test 可以自動單元測試并生產覆蓋率報表、也有 go get 套件管理工具(雖然沒有版本功能),不過都很實用,而且也不需要加上分號( ; ),真要說不好的地方??大概就是強迫你花括號不能換行放吧(沒錯,我就是花括號會換行放的人)。

當我在撰寫這份文件的時候 我會先假設你有一定的基礎 ,你可以先閱讀下列的手冊,他們都很不錯。

你能夠在PHP 里面想建立一個變數的時候就直接建立,夭壽贊,是嗎?

蒸蚌!那么Golang 呢?在Golang 中變數分為幾類:「新定義」、「預先定義」、「自動新定義」、「覆蓋」。讓我們來看看范例:

在PHP中你會很常用到 echo 來顯示文字,像這樣。

然而在Golang中你會需要 fmt 套件,關于「什么是套件」的說明你可以在文章下述了解。

這很簡單,而且兩個語言的用法相差甚少,下面這是PHP:

只是Golang 稍微聒噪了一點,你必須在函式后面宣告他最后會回傳什么資料型別。

在PHP 中你要回傳多個資料你就會用上陣列,然后將資料放入陣列里面,像這樣。

然而在Golang 中你可以不必用到一個陣列,函式可以一次回傳多個值:

兩個語言的撰寫方式不盡相同。

主要是PHP 的陣列能做太多事情了,所以在PHP 里面要儲存什么用陣列就好了。

在Golang里??沒有這么萬能的東西,首先要先了解Golang中有這些型態: array , slice , map , interface ,

你他媽的我到底看了三洨,首先你要知道Golang是個強型別語言,意思是你的陣列中 只能有一種型態 ,什么意思?當你決定這個陣列是用來擺放字串資料的時候,你就只能在里面放字串。沒有數值、沒有布林值,就像你沒有女朋友一樣。

先撇開PHP 的「萬能陣列」不管,Golang 中的陣列既單純卻又十分腦殘,在定義一個陣列的時候,你必須給他一個長度還有其內容存放的資料型態,你的陣列內容不一定要填滿其長度,但是你的陣列內容不能超過你當初定義的長度。

切片??這聽起來也許很奇怪,但是你確實可以「切」他,讓我們先談談「切片」比起「陣列」要好在哪里:「你不用定義其最大長度,而且你可以直接賦予值」,沒了。

我們剛才有提到你可以「切」他,記得嗎?這有點像是PHP中的 array_slice() ,但是Golang直接讓Slice「內建」了這個用法,其用法是: slice[開始:結束] 。

在PHP中倒是沒有那么方便,在下列PHP范例中你需要不斷地使用 array_slice() 。

你可以把「映照」看成是一個有鍵名和鍵值的陣列,但是記住:「你需要事先定義其鍵名、鍵值的資料型態」,這仍限制你沒辦法在映照中存放多種不同型態的資料。

在Golang里可就沒這么簡單了,你需要先用 make() 宣告 map 。

也許你不喜歡「接口」這個詞,但用「介面」我怕會誤導大眾,所以,是的,接下來我會繼續稱其為「接口」。還記得你可以在PHP 的關聯陣列里面存放任何型態的資料嗎,像下面這樣?

現在你有福了!正因為Golang中的 interface{} 可以接受任何內容,所以你可以把它拿來存放任何型態的資料。

有時候你也許會有個不定值的變數,在PHP 里你可以直接將一個變數定義成字串、數值、空值、就像你那變心的女友一樣隨時都在變。

在Golang中你必須給予變數一個指定的資料型別,不過還記得剛才提到的:「Golang中有個 interface{} 能夠 存放任何事物 」嗎( 雖然也不是真的任何事物啦?? )?

當我們程式中不需要繼續使用到某個資源或是發生錯誤的時候,我們索性會將其關閉或是拋棄來節省資源開銷,例如PHP 里的讀取檔案:

在Golang中,你可以使用 defer 來在函式結束的時候自動執行某些程式(其執行方向為反向)。所以你就不需要在函式最后面結束最前面的資源。

defer 可以被稱為「推遲執行」,實際上就是在函式結束后會「反序」執行的東西,例如你按照了這樣的順序定義 defer : A-B-C-D ,那么執行的順序其實會是 D-C-B-A ,這用在程式結束時還蠻有用的,讓我們看看Golang如何改善上述范例。

這東西很邪惡,不是嗎?又不是在寫BASIC,不過也許有時候你會在PHP 用上呢。但是拜托,不要。

Golang中僅有 for 一種回圈但卻能夠達成 foreach 、 while 、 for 多種用法。普通 for 回圈寫法在兩個語言中都十分相近。

在Golang請記得:如果你的 i 先前并不存在,那么你就需要定義它,所以下面這個范例你會看見 i := 0 。

在PHP里, foreach() 能夠直接給你值和鍵名,用起來十分簡單。

Golang里面雖然僅有 for() 但卻可以使用 range 達成和PHP一樣的 foreach 方式。

一個 while(條件) 回圈在PHP里面可以不斷地執行區塊中的程式,直到 條件 為 false 為止。

在Golang里也有相同的做法,但仍是透過 for 回圈,請注意這個 for 回圈并沒有任何的分號( ; ),而且一個沒有條件的 for 回圈會一直被執行。

PHP中有 do .. while() 回圈可以先做區塊中的動作。

在Golang中則沒有相關函式,但是你可以透過一個無止盡的 for 回圈加上條件式來讓他結束回圈。

要是你真的希望完全符合像是PHP那樣的設計方式,或者你可以在Golang中使用很邪惡的 goto 。

在PHP中我們可以透過 date() 像這樣取得目前的日期。

在Golang就稍微有趣點了,因為Golang中并不是以 Y-m-d 這種格式做為定義,而是 1 、 2 、 3 ,這令你需要去翻閱文件,才能夠知道 1 的定義是代表什么。

俗話說:「爆炸就是藝術」,可愛的PHP用詞真的很大膽,像是: explode() (爆炸)、 die() (死掉),回歸正傳,如果你想在PHP里面將字串切割成陣列,你可以這么做。

簡單的就讓一個字串給「爆炸」了,那么Golang 呢?

對了,記得引用 strings 套件。

這真的是很常用到的功能,就像物件一樣有著鍵名和鍵值,在PHP 里面你很簡單的就能靠陣列(Array)辦到。

真是太棒了,那么Golang呢?用 map 是差不多啦。如果有必要的話,你可以稍微復習一下先前提到的「多資料儲存型態-Stores」。

你很常會在PHP里面用 isset() 檢查一個索引是否存在,不是嗎?

在Golang里面很簡單的能夠這樣辦到(僅適用于 map )。

指針(有時也做參照)是一個像是「變數別名」的方法,這種方法讓你不用整天覆蓋舊的變數,讓我們假設 A = 1; B = A; 這個時候 B 會復制一份 A 且兩者不相干,倘若你希望修改 B 的時候實際上也會修改到 A 的值,就會需要指針。

指針比起復制一個變數,他會建立一個指向到某個變數的記憶體位置,這也就是為什么你改變指針,實際上是在改變某個變數。

在Golang你需要用上 * 還有 符號。

有些時候你會回傳一個陣列,這個陣列里面可能有資料還有錯誤代號,而你會用條件式判斷錯誤代號是否非空值。

在Golang中函式可以一次回傳多個值。為此,你不需要真的回傳一個陣列,不過要注意的是你將會回傳一個屬于 error 資料型態的錯誤,所以你需要引用 errors 套件來幫助你做這件事。

該注意的是Golang沒有 try .. catch ,因為 Golang推薦這種錯誤處理方式 ,你應該在每一次執行可能會發生錯誤的程式時就處理錯誤,而非后來用 try 到處包覆你的程式。

在 if 條件式里宣告變數會讓你只能在 if 內部使用這個變數,而不會污染到全域范圍。

也許你在PHP中更常用的會是 try .. catch ,在大型商業邏輯時經常看見如此地用法,實際上這種用法令人感到聒噪(因為你會需要一堆 try 區塊):

Golang中并沒有 try .. catch ,實際上Golang也 不鼓勵這種行為 (Golang推薦逐一處理錯誤的方式),倘若你真想辦倒像是捕捉異常這樣的方式,你確實可以使用Golang中另類處理錯誤的方式(可以的話盡量避免使用這種方式): panic() , recover() , defer 。

你可以把 panic() 當作是 throw (丟出錯誤),而這跟PHP的 exit() 有87%像,一但你執行了 panic() 你的程式就會宣告而終,但是別擔心,因為程式結束的時候會呼叫 defer ,所以我們接下來要在 defer 停止 panic() 。

關于 defer 上述已經有提到了,他是一個反向執行的宣告,會在函式結束后被執行,當你呼叫了 panic() 結束程式的時候,也就會開始執行 defer ,所以我們要在 defer 內使用 recover() 讓程式不再繼續進行結束動作,這就像是捕捉異常。

recover() 可以看作 catch (捕捉),我們要在 defer 里面用 recover() 解決 panic() ,如此一來程式就會回歸正常而不會被結束。

還記得在PHP里要引用一堆檔案的日子嗎?到處可見的 require() 或是 include() ?到了Golang這些都不見了,取而代之的是「套件(Package)」。現在讓我們來用PHP解釋一下。

這看起來很正常對吧?但假設你有一堆檔案,這馬上就成了 Include Hell ,讓我們看看Golang怎么透過「套件」解決這個問題。

「 蛤???殺小??? 」你可能如此地說道。是的, main.go 中除了引用 fmt 套件( 為了要輸出結果用的套件 )之外完全沒有引用到 a.go 。

「 蛤???殺小?????? 」你仿佛回到了幾秒鐘前的自己。

既然沒有引用其他檔案,為什么 main.go 可以輸出 foo 呢?注意到了嗎, 兩者都是屬于 main 套件 ,因此 他們共享同一個區域 ,所以接下來要介紹的是什么叫做「套件」。

套件是每一個 .go 檔案都必須聲明在Golang原始碼中最開端的東西,像下面這樣:

這意味著目前的檔案是屬于 main 套件( 你也可以依照你的喜好命名 ),那么要如何讓同個套件之間的函式溝通呢?

接著是Golang;注意!你不需要引用任何檔案,因為下列兩個檔案同屬一個套件。

一個由「套件」所掌握的世界,比起PHP的 include() 和 require() 還要好太多了,對嗎?

在Golang 中沒有引用單獨檔案的方式,你必須匯入一整個套件,而且你要記住:「一定你匯入了,你就一定要使用它」,像下面這樣。

假如你不希望使用你匯入的套件,你只是為了要觸發那個套件的 main() 函式而引用的話??,那么你可以在前面加上一個底線( _ )。

如果你的套件出現了名稱沖突,你可以在套件來源前面給他一個新的名稱。

現在你知道可以匯入套件了,那么什么是「匯出」?同個套件內的函式還有共享變數確實可以直接用,但那 并不表示可以給其他套件使用 ,其方法取決于 函式/變數的「開頭大小寫」 。

是的。 Golang依照一個函式/變數的開頭大小寫決定這個東西是否可供「匯出」 。

這用在區別函式的時候格外有用,因為小寫開頭的任何事物都是不供匯出的,反之,大寫開頭的任何事物都是用來匯出供其他套件使用的。

一開始可能會覺得這是什么奇異的規定,但寫久之后,你就能發現比起JavaScript和Python以「底線為開頭的命名方式」還要來得更好;比起成天宣告 public 、 private 、 protected 還要來得更快。

在Golang 中沒有類別,但有所謂的「建構體(Struct)」和「接口(Interface)」,這就能夠滿足幾乎所有的需求了,這也是為什么我認為Golang 很簡潔卻又很強大的原因。

讓我們先用PHP 建立一個類別,然后看看Golang 怎么解決這個問題。

雖然Golang沒有類別,但是「建構體(Struct)」就十分地堪用了,首先你要知道在Golang中「類別」的成員還有方法都是在「類別」外面所定義的,這跟PHP在類別內定義的方式有所不同,在Golang中還有一點,那就是他們沒有 public 、 private 、 protected 的種類。

在PHP中,當有一個類別被 new 的時候會自動執行該類別內的建構子( __construct() ),通常你會用這個來初始化一些類別內部的值。

但是在Golang 里因為沒有類別,也就沒有建構子,不巧的是建構體本身也不帶有建構子的特性,這個時候你只能自己在外部建立一個建構用函式。

讓我們假設你有兩個類別,你會把其中一個類別傳入到另一個類別里面使用,廢話不多說!先上個PHP 范例(為了簡短篇幅我省去了換行)。

在Golang中你也有相同的用法,但是請記得:「 任何東西都是在「類別」外完成建構的 」。

在PHP 中沒有相關的范例,這部分會以剛才「嵌入」章節中的Golang 范例作為解說對象。

你可以看見Golang在進行 Foo 嵌入 Bar 的時候,會自動將 Foo 的成員暴露在 Bar 底下,那么假設「雙方之間有相同的成員名稱」呢?

這個時候被嵌入的成員就會被「遮蔽」,下面是個實際范例,還有你如何解決遮蔽問題:

雖然都是呼叫同一個函式,但是這個函式可以針對不同的資料來源做出不同的舉動,這就是多形。你也能夠把這看作是:「訊息的意義由接收者定義,而不是傳送者」。

目前PHP 中沒有真正的「多形」,不過你仍可以做出同樣的東西。

嗯??那么Golang呢?實際上更簡單而且更有條理了,在Golang中有 interface 可以幫忙完成這個工作。

如果你對Interface還不熟悉,可以試著查看「 解釋Golang中的Interface到底是什么 」文章。

謝謝你看到這里,可惜這篇文章卻沒有說出Golang 最重要的賣點:「Goroutine」和「Channel」

go語言中關于defer的注意事項

舉個例子,如下

答案

解析:

defer函數在處理Panic() 和Recover()時的應用

panic 函數是內置的go函數,它 終止 go程序的當前流程并開始 panicking , recover 函數也是內置的一個go函數,允許你收回處理那些使用了 panic 函數的 goroutine 的控制權

來個案例

網站標題:go語言中的defer Go語言中的Handle
網站地址:http://m.kartarina.com/article8/hjipop.html

成都網站建設公司_創新互聯,為您提供網站策劃移動網站建設服務器托管手機網站建設品牌網站制作

廣告

聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯

成都定制網站網頁設計
主站蜘蛛池模板: 免费无码一区二区| 亚洲中文字幕无码一区| 亚洲毛片无码专区亚洲乱| 日韩一区二区三区无码影院| 精品国产a∨无码一区二区三区| 亚洲日韩av无码中文| 成人av片无码免费天天看| 亚洲AV综合色区无码二区爱AV| 久久久无码精品午夜| 亚洲AV成人无码久久精品老人 | 中文人妻无码一区二区三区| 99无码人妻一区二区三区免费| 国产亚洲情侣一区二区无码AV| 亚洲最大中文字幕无码网站| 亚洲国产精品成人精品无码区在线| 无码精品久久一区二区三区| 国产成人无码精品一区二区三区 | 日韩精品无码免费专区午夜不卡| 亚洲中文无码a∨在线观看| 中文字幕无码精品亚洲资源网| 中文字幕韩国三级理论无码| 无码人妻精品一区二区在线视频| 午夜无码一区二区三区在线观看| 亚洲精品无码高潮喷水A片软| 无码无遮挡又大又爽又黄的视频 | 亚洲AV永久无码精品一福利| 国产精品亚洲а∨无码播放| 国产在线无码精品电影网| 免费无码国产V片在线观看| 蜜桃AV无码免费看永久| 少妇无码AV无码专区在线观看 | 精品无码日韩一区二区三区不卡 | 永久免费av无码网站大全| 亚洲天堂2017无码中文| 在线看无码的免费网站| 久热中文字幕无码视频| 亚洲精品无码成人| 妖精色AV无码国产在线看| 国产午夜激无码av毛片| 国产成人无码a区在线视频| 亚洲中文字幕无码爆乳av中文|