02 Golang 基礎簡介

来源:https://www.cnblogs.com/cdc1216/archive/2023/03/05/17181140.html
-Advertisement-
Play Games

一、標識符 標識符是指定義的具有特殊意義的詞,例如變數、常量、函數名等等,任何一門語言中都對自己的標識符有特殊定義的規則。在 Go 語言中,標識符由字母數字和下劃線組成,並且只能以字母和下劃線開頭,例如: 數字、字母和下劃線組成:123、abc _ 只能以字母和下劃線開頭:abc123、_sysVa ...


一、標識符

標識符是指定義的具有特殊意義的詞,例如變數、常量、函數名等等,任何一門語言中都對自己的標識符有特殊定義的規則。在 Go 語言中,標識符由字母數字和下劃線組成,並且只能以字母和下劃線開頭,例如:

  • 數字、字母和下劃線組成:123abc _

  • 只能以字母和下劃線開頭:abc123_sysVar123abc

  • 標識符區分大小寫:nameNameNAME

二、關鍵字和保留字

關鍵字和保留字是指編程語言中預先定義好的具有特殊含義的標識符。 關鍵字和保留字都不建議用作變數名,會引起混亂和衝突。

1. GO中的關鍵字

    break        default      func         interface    select
    case         defer        go           map          struct
    chan         else         goto         package      switch
    const        fallthrough  if           range        type
    continue     for          import       return       var

2. GO中的保留字

 Constants:    true  false  iota  nil
​
     Types:    int  int8  int16  int32  int64  
               uint  uint8  uint16  uint32  uint64  uintptr
               float32  float64  complex128  complex64
               bool  byte  rune  string  error
​
  Functions:   make  len  cap  new  append  copy  close  delete
               complex  real  imag
               panic  recover

三、命名規範

由於Go語言是一門區分大小寫的語言,因此Go從語法層面進行了以下限定:任何需要對外暴露的名字必須以大寫字母開頭,不需要對外暴露的則應該以小寫字母開頭。

當命名(包括常量、變數、類型、函數名、結構欄位等等)以一個大寫字母開頭,如:GetUserName,那麼使用這種形式的標識符的對象就可以被外部包的代碼所使用(程式需要先導入這個包),這被稱為導出(類似面向對象語言中的公共屬性); 命名如果以小寫字母開頭,則對包外是不可見的,但是他們在整個包的內部是可見並且可用的(類似面向對象語言中的私有屬性 )。

Go語言中各類情形的建議命名規則如下:

  • 變數命名

變數名稱一般遵循駝峰法,首字母根據訪問控制原則大寫或者小寫

1 var userName string
2 var isExist bool
  • 常量命名

常量均需使用全部大寫字母組成,並使用下劃線分詞

const SITE_URL = "http://www.chendacheng.com"
  • 結構體命名

採用駝峰命名法,首字母根據訪問控制大寫或者小寫

1 type UserInfo struct {
2     Name string,
3     age  int,
4 }
  • 介面命名

命名規則基本和上面的結構體類型,單個函數的結構名以er作為尾碼

1 type Reader interface {
2      Read(p []byte) (n int, err error)
3 }
  • 錯誤處理

錯誤處理的原則就是不能丟棄任何有返回err的調用,不要使用_丟棄,必須全部處理。接收到錯誤,要麼返回err,或者使用log記錄下來儘早return。一旦有錯誤發生,馬上返回,儘量不要使用panic,除非你知道你在做什麼,錯誤描述如果是英文必須為小寫,不需要標點結尾,採用獨立的錯誤流進行處理。

1 if err != nil {
2     // 錯誤處理
3     return // 或者繼續
4 }
5 // 正常代碼
  • 包命名

儘量保持和目錄保持一致,採取有意義的包名,簡短且不要和標準庫不要衝突。包名應該為小寫單詞,不要使用下劃線或者混合大小寫。

1 package dao
2 package service
3 package main
  • 文件命名

儘量採取簡短且有意義的文件名,應該為小寫單詞,使用下劃線分隔各個單詞。

1 customer_dao.go
2 user_manage.go
  • 單元測試

單元測試文件名要以 _test.go結尾,測試文件中的測試用例的函數名稱必須以 Test 開頭。

四、變數

變數的作用是存儲數據,不同的變數保存的數據類型可能會不一樣。Go 語言中的每一個變數都有自己的類型,變數必須經過聲明才能開始使用,且同一作用域內不支持重覆聲明

1. 變數的作用域

1.1 全局變數和局部變數

變數可以定義在函數內部(函數外的每個語句都必須以關鍵字開始,如:varconstfunc等),也可以定義在函數內部。定義在函數外部的變數稱為 全局變數,定義在函數內部的變數稱為 局部變數 。在 GO 語言中,定義的局部變數必須使用,否則編譯代碼的時候將不被通過,定義的全局變數可以不使用。

1 package main
2 3 var name string = "cdc" // 定義一個全局變數
4 5 func main() {
6 7 }

直接編譯通過:

1 package main
2 3 func main() {
4     name := "cdc"  // 聲明並初始化了一個局部變數,但是未使用
5 }

直接編譯未通過,報錯:

1.2 作用域

  • 函數內可以使用全局的變數,但是在全局無法使用局部的變數

 1 var name = "cdc"
 2  3 func main() {
 4     fmt.Printf("%v\n", name)  // cdc
 5 }
 6 func demo() {
 7     var name = "cdc"
 8 }
 9 10 func main() {
11     fmt.Printf("%v\n", name) // undefined: namet
12 13 }
  • 代碼執行時,先從函數內部尋找局部變數,找不到再去找全局的變數

 1 package main
 2  3 import "fmt"
 4  5 var name = "cdc"
 6 var age = 22
 7  8 func main() {
 9 10     var name = "ctt"
11     
12 13     fmt.Printf("%v\n", name) // ctt
14     fmt.Printf("%v\n", age)  // 22
15 }

2. 變數的聲明

2.1 標準聲明方式

變數聲明以關鍵字 var 開頭,變數類型放在變數的後面,行尾無需分號。

1 var name string
2 var age int
3 var isOk bool

2.2 批量聲明

1 var (
2     a string
3     b int
4     c bool
5     d float32
6 )

註意:在沒有初始化變數之前,不同數據類型的變數會有一個預設值,值為該數據類型對應的0值:

 1 package main
 2  3 import "fmt"
 4  5 func main() {
 6     var (
 7         a string
 8         b int
 9         c bool
10         d float32
11     )
12 13     fmt.Println(a) // ""
14     fmt.Println(b) // 0
15     fmt.Println(c) // false
16     fmt.Println(d) // 0
17 }

3. 變數初始化

3.1 標準初始化格式

1 var name string = "cdc"
2 var age int = 18
3 4 // 一次聲明多個變數
5 var age, isOk = 18, true

3.2 類型推導

有時候我們會將變數的類型省略,這個時候編譯器會根據等號右邊的值來推導變數的類型完成初始化

 1 package main
 2  3 import "fmt"
 4  5 func main() {
 6     var name = "cdc" // 編譯器會根據 “cdc” 推導出變數 name 是一個字元串類型
 7     var age = 18
 8  9     fmt.Printf("%T\n", name) // string
10     fmt.Printf("%T\n", age)  // int
11 }

3.3 短變數聲明

短變數聲明方式只能用於函數內部

 1 package main
 2  3 import "fmt"
 4  5 // 全局變數m
 6 var m = 100
 7  8 func main() {
 9     n := 10
10     m := 200 // 此處聲明局部變數m
11     fmt.Println(m, n)
12 }

3.4 匿名變數

對於聲明的局部變數必須要使用,否則編譯無法通過。如果想要忽略某個值,我們可以使用 匿名變數 來接收該值 。匿名變數用一個下劃線表示,它不占用命名空間,不會分配記憶體,所以匿名變數之間不存在重覆聲明,例如:

 1 package main
 2  3 import "fmt"
 4  5 func function1() (string, int) {
 6     return "cdc", 18
 7 }
 8  9 func main() {
10     var name, _ = function1()
11     fmt.Printf("My name is %s", name)
12 }

匿名變數 _ 並未使用,但是編譯可以通過

五、常量

常量是指恆定不變的值,多用於定義程式運行期間不會改變的那些值,一旦定義了常量後就無法修改。

1. 標準聲明格式

1 const PI = 3.1415
2 const E = 2.7182

2. 批量聲明

1 const (
2     STATUS_OK = 200
3     NOT_FOUND = 404
4 )

批量聲明常量時,如果某一行聲明之後沒有賦值,那麼後面的常量就預設和上一行一致

 1 package main
 2  3 import "fmt"
 4  5 const (
 6     n1 = 100
 7     n2 = 200
 8     n3
 9     n4
10 )
11 12 func main() {
13     fmt.Printf("n1:%v\n", n1)
14     fmt.Printf("n2:%v\n", n2)
15     fmt.Printf("n3:%v\n", n3)
16     fmt.Printf("n4:%v\n", n4)
17 }

編譯執行結果如下,n3n4 的值都為 200:

3. iota

iota 是go語言的常量計數器,只能在常量的表達式中使用。iotaconst 關鍵字出現時將被重置為0,const 中每新增一行常量聲明將使 iota 計數一次。可以直接理解 iota 其實就是每一行代碼的索引值。

  • 示例1:

 1 package main
 2  3 import "fmt"
 4  5 const (
 6     a1 = iota
 7     a2 = iota
 8     a3 = iota
 9     a4 = iota
10 )
11 12 func main() {
13     fmt.Printf("a1:%d\n", a1)
14     fmt.Printf("a2:%d\n", a2)
15     fmt.Printf("a3:%d\n", a3)
16     fmt.Printf("a4:%d\n", a4)
17 }

分析:出現了 const 關鍵字,所以 a1 對應的 iota 的值為 0;後面每新增一行常量的聲明,iota 的值就累加1,所以最後列印的結果為:

  • 示例2,省略 iota

 1 package main
 2  3 import "fmt"
 4  5 const (
 6     b1 = iota
 7     b2
 8     b3
 9     b4
10 )
11 12 func main() {
13     fmt.Printf("b1:%d\n", b1)
14     fmt.Printf("b2:%d\n", b2)
15     fmt.Printf("b3:%d\n", b3)
16     fmt.Printf("b4:%d\n", b4)
17 }

分析:出現了 const 關鍵字,所以 b1 對應的 iota 的值為 0;由於常量批量聲明的規則,當某一行聲明之後沒有賦值,那麼後面的常量就預設和上一行一致,所以理論上 b2 的值應該也為 iota,每新增一行常量的聲明,iota 的值就累加1,所以 b2 的值應該為1,以此類推,最後列印的結果為:

  • 示例3,使用 _ 跳過某些值:

 1 package main
 2  3 import "fmt"
 4  5 func main() {
 6     const (
 7         n1 = iota
 8         n2
 9         _
10         n4
11     )
12 13     fmt.Printf("n1: %d\n", n1)
14     fmt.Printf("n2: %d\n", n2)
15     fmt.Printf("n4: %d\n", n4)
16 }

分析:出現了 const 關鍵字,所以 n1 對應的 iota 的值為 0;由於常量批量聲明的規則,當某一行聲明之後沒有賦值,那麼後面的常量就預設和上一行一致,所以理論上 n2 的值應該也為 iota,每新增一行常量的聲明,iota 的值就累加1,所以 n2 的值應該為1;雖然匿名變數會被跳過,但是也是作為一個常量聲明的,也會遵循只要新增一行常量聲明 iota 就累加1的規則,所以匿名變數對應的值應該是2,以此類推,最後編譯列印的結果為:

  • 示例4,iota 聲明中間插隊:

 1 package main
 2  3 import "fmt"
 4  5 func main() {
 6     const (
 7         n1 = iota
 8         n2 = 100
 9         n3 = iota
10         n4
11     )
12 13     fmt.Printf("n1: %d\n", n1)
14     fmt.Printf("n3: %d\n", n2)
15     fmt.Printf("n4: %d\n", n4)
16 }

分析:出現了 const 關鍵字,所以 n1 對應的 iota 的值為 0;雖然 n2 沒有使用到 iota,但是 iota 是對當前批量聲明的常量做統計的,只要新增了一行常量聲明,值就累加 1 ,因此聲明 n2 時,iota 還是會加 1,以此類推,最後編譯列印的結果為:

  • 示例5,多個iota定義在一行

 1 package main
 2  3 import "fmt"
 4  5 func main() {
 6     const (
 7         n1, n2 = iota + 1, iota + 2
 8         n3, n4 = iota + 1, iota + 2
 9     )
10 11     fmt.Printf("n1: %d\n", n1)
12     fmt.Printf("n2: %d\n", n2)
13     fmt.Printf("n3: %d\n", n3)
14     fmt.Printf("n4: %d\n", n4)
15 }

分析:只要每新增了一行常量聲明,iota 值就累加 1 ,但是 n1n2 是在一行聲明的,所以對於 n1n2iota 的值都為 0;到聲明 n3n4 的時候才是新增了一行聲明,這時的 iota 的值才會累加 1,編譯運行的結果如下:

  • 示例6,使用 iota 定義數量級

1 const (
2         _  = iota
3         KB = 1 << (10 * iota)
4         MB = 1 << (10 * iota)
5         GB = 1 << (10 * iota)
6         TB = 1 << (10 * iota)
7         PB = 1 << (10 * iota)
8 )

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

-Advertisement-
Play Games
更多相關文章
  • 1. 編譯閾值 1.1. 一旦代碼執行到一定次數,就達到了它的編譯閾值,編譯器就會認為它有足夠的信息來編譯代碼 1.2. 在當前的JVM中,優化閾值的意義不大 1.2.1. 從JDK 7以及更早期遺留下來的 1.3. -XX:CompileThreshold=N 1.3.1. 當禁用分層編譯時有效 ...
  • 0. RSocket 簡介 採用二進位點對點數據傳輸,主要應用於分散式架構之中,是一種基於Reactive Stream規範標準實現的新的通信協議。 參考阿裡雲開發者社區的介紹 相關文檔和資料: RSocket By Example rsocket-java 原生庫例子 Spring RSocket ...
  • 環境 odoo-14.0.post20221212.tar context用法總結 獲取上下文 >>> self.env.context # 返回字典數據,等價於 self._context {'lang': 'en_US', 'tz': 'Europe/Brussels'} >>> self._c ...
  • Qt 學習筆記全系列傳送門: 【本章】Qt 學習筆記 - 第一章 - 快速開始、信號與槽 Qt 學習筆記 - 第二章 - 添加圖片、佈局、界面切換 1、Qt 工程創建 使用 Qt Creator 創建 Qt 工程,不能包含中文目錄 2、工程文件(pro 文件) # # # Project creat ...
  • 項目實現01 1.功能01-搭建Vue前端工程 1.1需求分析 使用Vue3的腳手架vue-cli工具,創建ssm的前端項目基礎開發環境 Vue-cli主要的功能是自動生成Vue的項目模板,提高開發效率 1.2代碼實現 1.2.1搭建vue工程 以下命令使用管理員許可權 (1)先下載node.js L ...
  • VL45 非同步FIFO 很經典的手撕題,這道題要求產生的格雷碼要在本時鐘域中打一拍,其實不打也沒關係。 主要要記住 1、bin2gray的方法:右移一位與移位前異或; 2、格雷碼比較方法:空:讀指針格雷碼和寫指針同步過來的格雷碼相同;滿:寫指針格雷碼高兩位與讀指針同步過來的格雷碼正好相反,低位相同。 ...
  • 職工工資管理系統程式中新增臨時員工的功能 這個職工工資管理系統基本功能都十分完善,代碼也都比較簡潔,運行效率也很高,沒有什麼直接能改的功能,所以我基於現在許多大學生找兼職、臨時工的現象,增添了臨時員工的工資系統,可以方便公司、企業等更加完善公司的財務系統。也方便公司為臨時員工發放工資。 代碼段一是原 ...
  • 雙指針 同向雙指針 能夠實現跳躍尋找,適用於尋找含有某一特性區間,比如最長相同區間,最長不重覆區間 不重覆區間可以用一個數組t[N]來表示,如果其中元素大於1,說明有重覆 int res=0,j=0; for(int i=0;i<n;i++) { t[a[i]]++;//記錄個數 while(j<i ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...