Go語(yǔ)言中nil判斷引起的問題詳析
前言
代碼封裝是百干不厭的事,但有時(shí)候封裝會(huì)導(dǎo)致一些問題。本文記錄了個(gè)人在封裝 http 請(qǐng)求時(shí)遇到的一個(gè)和 nil 判斷有關(guān)的問題。
nil 是什么
在 Go 語(yǔ)言中,布爾類型的零值(初始值)為 false,數(shù)值類型的零值為 0,字符串類型的零值為空字符串"",而指針、切片、映射、通道、函數(shù)和接口的零值則是 nil。
nil 內(nèi)置的一個(gè)變量,用來代表空值,且只有指針、channel、方法、接口、map 和切片可以被賦值為 nil。
有過其他編程語(yǔ)言開發(fā)經(jīng)驗(yàn)的開發(fā)者也許會(huì)把 nil 看作其他語(yǔ)言中的 null(NULL),其實(shí)這并不是完全正確的,因?yàn)镚o語(yǔ)言中的 nil 和其他語(yǔ)言中的 null 有很多不同點(diǎn)。
buildin/buildin.go:
// nil is a predeclared identifier representing the zero value for a
// pointer, channel, func, interface, map, or slice type.
var nil Type // Type must be a pointer, channel, func, interface, map, or slice type// Type is here for the purposes of documentation only. It is a stand-in
// for any Go type, but represents the same type for any given function
// invocation.
type Type int
問題代碼
下面的代碼是我對(duì) http.Post 方法的封裝
func (r *Request) Post(endpoint string, params *url.Values, body io.Reader, headers map[string]string, cookies map[string]string) (resp *http.Response, err error) {
url := fmt.Sprintf("%s%s", r.BaseURL, endpoint)
var req *http.Request
req, err = http.NewRequest(http.MethodPost, url, body)
if err != nil {
return
}
r.setRequest(req, params, headers, cookies)
resp, err = r.Client.Do(req)
return
}
然后像下面這樣使用的時(shí)候:
var body *bytes.Reader body = nil resp, err = req.Post(endpoint, nil, body, nil, nil)
這時(shí)會(huì)出現(xiàn)空指針的錯(cuò)誤,最終經(jīng)過漫長(zhǎng)的排查發(fā)現(xiàn)是在 http.NewRequest 里出現(xiàn)的空指針錯(cuò)誤:
錯(cuò)誤分析
指針和接口的底層實(shí)現(xiàn)有兩部分:data 和 type。當(dāng)指針和接口被顯式地賦值為 nil 時(shí),data 和 type 同時(shí)為 nil,但是將一個(gè) type 不為 nil 但 data 為 nil 的值賦值給指針或接口時(shí),再與 nil 作比較的結(jié)果則是 false。
修改代碼
使用 reflect.ValueOf(body).IsNil() 判斷 body 是否為空:
func (r *Request) Post(endpoint string, params *url.Values, body io.Reader, headers map[string]string, cookies map[string]string) (resp *http.Response, err error) {
url := fmt.Sprintf("%s%s", r.BaseURL, endpoint)
var req *http.Request
if reflect.ValueOf(body).IsNil() {
req, err = http.NewRequest(http.MethodPost, url, nil)
} else {
req, err = http.NewRequest(http.MethodPost, url, body)
}
if err != nil {
return
}
r.setRequest(req, params, headers, cookies)
resp, err = r.Client.Do(req)
return
}
總結(jié)
到此這篇關(guān)于Go語(yǔ)言中nil判斷引起問題的文章就介紹到這了,更多相關(guān)Go nil判斷問題內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
一文帶你了解Go語(yǔ)言中time包的時(shí)間常用操作
在日常開發(fā)中,我們避免不了時(shí)間的使用,我們可能需要獲取當(dāng)前時(shí)間,然后格式化保存,也可能需要在時(shí)間類型與字符串類型之間相互轉(zhuǎn)換等。本文將會(huì)對(duì)?Go?time?包里面的常用函數(shù)和方法進(jìn)行介紹,需要的可以參考一下2022-12-12
Go語(yǔ)言到底有沒有引用傳參(對(duì)比 C++ )
這篇文章主要介紹了Go 到底有沒有引用傳參(對(duì)比 C++ ),需要的朋友可以參考下2017-09-09
關(guān)于go語(yǔ)言載入json可能遇到的一個(gè)坑
Go 語(yǔ)言從新手到大神,每個(gè)人多少都會(huì)踩一些坑,那么下面這篇文章主要給大家介紹了關(guān)于go語(yǔ)言載入json可能遇到的一個(gè)坑,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起看看吧。2017-07-07
Golang利用compress/flate包來壓縮和解壓數(shù)據(jù)
在處理需要高效存儲(chǔ)和快速傳輸?shù)臄?shù)據(jù)時(shí),數(shù)據(jù)壓縮成為了一項(xiàng)不可或缺的技術(shù),Go語(yǔ)言的compress/flate包為我們提供了對(duì)DEFLATE壓縮格式的原生支持,本文將深入探討compress/flate包的使用方法,揭示如何利用它來壓縮和解壓數(shù)據(jù),并提供實(shí)際的代碼示例,需要的朋友可以參考下2024-08-08
Go語(yǔ)言使用組合的思想實(shí)現(xiàn)繼承
這篇文章主要為大家詳細(xì)介紹了在 Go 里面如何使用組合的思想實(shí)現(xiàn)“繼承”,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)Go語(yǔ)言有一定的幫助,需要的可以了解一下2022-12-12
Golang使用Channel組建高并發(fā)HTTP服務(wù)器
Golang 作為一門高效的語(yǔ)言,在網(wǎng)絡(luò)編程方面表現(xiàn)也非常出色,這篇文章主要介紹了如何使用 Golang 和 Channel 組建高并發(fā) HTTP 服務(wù)器,感興趣的可以了解一下2023-06-06

