鍍金池/ 教程/ GO/ 10.2 使用工廠方法創(chuàng)建結(jié)構(gòu)體實(shí)例
4.7 strings 和 strconv 包
13.6 啟動(dòng)外部命令和程序
?# 11.4 類型判斷:type-switch
12.1 讀取用戶的輸入
10.6 方法
12.2 文件讀寫(xiě)
13 錯(cuò)誤處理與測(cè)試
9.3 鎖和 sync 包
12.3 文件拷貝
?# 11.7 第一個(gè)例子:使用 Sorter 接口排序
?# 11.5 測(cè)試一個(gè)值是否實(shí)現(xiàn)了某個(gè)接口
6.4 defer 和追蹤
12.10 XML 數(shù)據(jù)格式
13.10 性能調(diào)試:分析并優(yōu)化 Go 程序
?# 11.1 接口是什么
2.2 Go 環(huán)境變量
2.6 安裝目錄清單
2.5 在 Windows 上安裝 Go
11.11 Printf 和反射
1.2 語(yǔ)言的主要特性與發(fā)展的環(huán)境和影響因素
9.0 包(package)
7.4 切片重組(reslice)
13.2 運(yùn)行時(shí)異常和 panic
10.2 使用工廠方法創(chuàng)建結(jié)構(gòu)體實(shí)例
12.8 使用接口的實(shí)際例子:fmt.Fprintf
2.4 在 Mac OS X 上安裝 Go
3.8 Go 性能說(shuō)明
7.2 切片
8.0 Map
3.1 Go 開(kāi)發(fā)環(huán)境的基本要求
5.6 標(biāo)簽與 goto
6.10 使用閉包調(diào)試
9.5 自定義包和可見(jiàn)性
4.3 常量
?# 11.2 接口嵌套接口
6.5 內(nèi)置函數(shù)
前言
10.8 垃圾回收和 SetFinalizer
2.8 Go 解釋器
13.7 Go 中的單元測(cè)試和基準(zhǔn)測(cè)試
6.8 閉包
4.9 指針
13.1 錯(cuò)誤處理
10.1 結(jié)構(gòu)體定義
5.1 if-else 結(jié)構(gòu)
6.6 遞歸函數(shù)
9.9 通過(guò) Git 打包和安裝
2.7 Go 運(yùn)行時(shí)(runtime)
10.7 類型的 String() 方法和格式化描述符
3.7 其它工具
9.6 為自定義包使用 godoc
11.12 接口與動(dòng)態(tài)類型
13.3 從 panic 中恢復(fù)(Recover)
10.3 使用自定義包中的結(jié)構(gòu)體
11.14 結(jié)構(gòu)體、集合和高階函數(shù)
3.6 生成代碼文檔
9.2 regexp 包
4.1 文件名、關(guān)鍵字與標(biāo)識(shí)符
?# 11.6 使用方法集與接口
7.0 數(shù)組與切片
7.1 聲明和初始化
12.11 用 Gob 傳輸數(shù)據(jù)
5.5 Break 與 continue
1.1 起源與發(fā)展
?# 11 接口(Interfaces)與反射(reflection)
6.9 應(yīng)用閉包:將函數(shù)作為返回值
4.2 Go 程序的基本結(jié)構(gòu)和要素
8.6 將 map 的鍵值對(duì)調(diào)
6.11 計(jì)算函數(shù)執(zhí)行時(shí)間
5.0 控制結(jié)構(gòu)
10.5 匿名字段和內(nèi)嵌結(jié)構(gòu)體
4.6 字符串
3.0 編輯器、集成開(kāi)發(fā)環(huán)境與其它工具
13.8 測(cè)試的具體例子
7.6 字符串、數(shù)組和切片的應(yīng)用
8.4 map 類型的切片
3.9 與其它語(yǔ)言進(jìn)行交互
7.3 For-range 結(jié)構(gòu)
9.7 使用 go install 安裝自定義包
6.0 函數(shù)
9.8 自定義包的目錄結(jié)構(gòu)、go install 和 go test
6.3 傳遞變長(zhǎng)參數(shù)
13.9 用(測(cè)試數(shù)據(jù))表驅(qū)動(dòng)測(cè)試
11.9 空接口
8.1 聲明、初始化和 make
6.2 函數(shù)參數(shù)與返回值
9.11 在 Go 程序中使用外部庫(kù)
3.3 調(diào)試器
4.5 基本類型和運(yùn)算符
?# 11.8 第二個(gè)例子:讀和寫(xiě)
12.5 用 buffer 讀取文件
總結(jié):Go 中的面向?qū)ο?/span>
11.10 反射包
12.7 用 defer 關(guān)閉文件
9.4 精密計(jì)算和 big 包
4.4 變量
6.1 介紹
13.4 自定義包中的錯(cuò)誤處理和 panicking
12.4 從命令行讀取參數(shù)
9.10 Go 的外部包和項(xiàng)目
8.3 for-range 的配套用法
3.5 格式化代碼
10.4 帶標(biāo)簽的結(jié)構(gòu)體
7.5 切片的復(fù)制與追加
?# 11.3 類型斷言:如何檢測(cè)和轉(zhuǎn)換接口變量的類型
5.4 for 結(jié)構(gòu)
4.8 時(shí)間和日期
2.3 在 Linux 上安裝 Go
12 讀寫(xiě)數(shù)據(jù)
6.12 通過(guò)內(nèi)存緩存來(lái)提升性能
9.1 標(biāo)準(zhǔn)庫(kù)概述
12.6 用切片讀寫(xiě)文件
10 結(jié)構(gòu)(struct)與方法(method)
8.5 map 的排序
12.9 JSON 數(shù)據(jù)格式
13.5 一種用閉包處理錯(cuò)誤的模式
3.2 編輯器和集成開(kāi)發(fā)環(huán)境
12.12 Go 中的密碼學(xué)
5.2 測(cè)試多返回值函數(shù)的錯(cuò)誤
6.7 將函數(shù)作為參數(shù)
8.2 測(cè)試鍵值對(duì)是否存在及刪除元素
3.4 構(gòu)建并運(yùn)行 Go 程序
2.1 平臺(tái)與架構(gòu)
5.3 switch 結(jié)構(gòu)

10.2 使用工廠方法創(chuàng)建結(jié)構(gòu)體實(shí)例

10.2.1 結(jié)構(gòu)體工廠

Go 語(yǔ)言不支持面向?qū)ο缶幊陶Z(yǔ)言中那樣的構(gòu)造子方法,但是可以很容易的在 Go 中實(shí)現(xiàn) “構(gòu)造子工廠”方法。為了方便通常會(huì)為類型定義一個(gè)工廠,按慣例,工廠的名字以 new 或 New 開(kāi)頭。假設(shè)定義了如下的 File 結(jié)構(gòu)體類型:

type File struct {
    fd      int     // 文件描述符
    name    string  // 文件名
}

下面是這個(gè)結(jié)構(gòu)體類型對(duì)應(yīng)的工廠方法,它返回一個(gè)指向結(jié)構(gòu)體實(shí)例的指針:

func NewFile(fd int, name string) *File {
    if fd < 0 {
        return nil
    }

    return &File{fd, name}
}

然后這樣調(diào)用它:

f := NewFile(10, "./test.txt")

在 Go 語(yǔ)言中常常像上面這樣在工廠方法里使用初始化來(lái)簡(jiǎn)便的實(shí)現(xiàn)構(gòu)造函數(shù)。

如果 File 是一個(gè)結(jié)構(gòu)體類型,那么表達(dá)式 new(File)&File{} 是等價(jià)的。

這可以和大多數(shù)面向?qū)ο缶幊陶Z(yǔ)言中笨拙的初始化方式做個(gè)比較:File f = new File(...)。

我們可以說(shuō)是工廠實(shí)例化了類型的一個(gè)對(duì)象,就像在基于類的OO語(yǔ)言中那樣。

如果想知道結(jié)構(gòu)體類型T的一個(gè)實(shí)例占用了多少內(nèi)存,可以使用:size := unsafe.Sizeof(T{})

如何強(qiáng)制使用工廠方法

通過(guò)應(yīng)用可見(jiàn)性規(guī)則參考4.2.1節(jié)9.5 節(jié)就可以禁止使用 new 函數(shù),強(qiáng)制用戶使用工廠方法,從而使類型變成私有的,就像在面向?qū)ο笳Z(yǔ)言中那樣。

type matrix struct {
    ...
}

func NewMatrix(params) *matrix {
    m := new(matrix) // 初始化 m
    return m
}

在其他包里使用工廠方法:

package main
import "matrix"
...
wrong := new(matrix.matrix)     // 編譯失?。╩atrix 是私有的)
right := matrix.NewMatrix(...)  // 實(shí)例化 matrix 的唯一方式

10.2.2 map 和 struct vs new() 和 make()

new 和 make 這兩個(gè)內(nèi)置函數(shù)已經(jīng)在第 7.2.4 節(jié)通過(guò)切片的例子說(shuō)明過(guò)一次。

現(xiàn)在為止我們已經(jīng)見(jiàn)到了可以使用 make() 的三種類型中的其中兩個(gè):

slices  /  maps / channels(見(jiàn)第 14 章)

下面的例子說(shuō)明了在映射上使用 new 和 make 的區(qū)別以及可能發(fā)生的錯(cuò)誤:

示例 10.4 new_make.go(不能編譯)

package main

type Foo map[string]string
type Bar struct {
    thingOne string
    thingTwo int
}

func main() {
    // OK
    y := new(Bar)
    (*y).thingOne = "hello"
    (*y).thingTwo = 1

    // NOT OK
    z := make(Bar) // 編譯錯(cuò)誤:cannot make type Bar
    (*z).thingOne = "hello"
    (*z).thingTwo = 1

    // OK
    x := make(Foo)
    x["x"] = "goodbye"
    x["y"] = "world"

    // NOT OK
    u := new(Foo)
    (*u)["x"] = "goodbye" // 運(yùn)行時(shí)錯(cuò)誤!! panic: assignment to entry in nil map
    (*u)["y"] = "world"
}

試圖 make() 一個(gè)結(jié)構(gòu)體變量,會(huì)引發(fā)一個(gè)編譯錯(cuò)誤,這還不是太糟糕,但是 new() 一個(gè)映射并試圖使用數(shù)據(jù)填充它,將會(huì)引發(fā)運(yùn)行時(shí)錯(cuò)誤! 因?yàn)?new(Foo) 返回的是一個(gè)指向 nil 的指針,它尚未被分配內(nèi)存。所以在使用 map 時(shí)要特別謹(jǐn)慎。

鏈接