国产精品久久久aaaa,日日干夜夜操天天插,亚洲乱熟女香蕉一区二区三区少妇,99精品国产高清一区二区三区,国产成人精品一区二区色戒,久久久国产精品成人免费,亚洲精品毛片久久久久,99久久婷婷国产综合精品电影,国产一区二区三区任你鲁

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

10種go語(yǔ)言編成中可能導(dǎo)致性能下降的壞實(shí)踐

馬哥Linux運(yùn)維 ? 來(lái)源:Teiva Harsanyi ? 作者:Teiva Harsanyi ? 2021-09-24 16:55 ? 次閱讀
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

本文總結(jié)了10種 go 語(yǔ)言編成中可能導(dǎo)致性能下降的壞實(shí)踐。有代碼潔癖的同學(xué)來(lái)自我檢查吧!

這篇文章主要講述了我在 Go 項(xiàng)目中見(jiàn)到過(guò)的常見(jiàn)錯(cuò)誤清單,順序無(wú)關(guān)。

未知的Enum

來(lái)看個(gè)簡(jiǎn)單的例子

typeStatusuint32

const(
StatusOpenStatus=iota
StatusClose
StatusUnknown
)

在上面的代碼中,使用iota創(chuàng)建了一個(gè)enum類型,分別代指下面的狀態(tài)信息:

StatusOpen=0
StatusClose=1
StatusUnknown=2

現(xiàn)在,我們假設(shè)Status是一個(gè) JSON 請(qǐng)求中被Marshalled / Unmarshalled的一個(gè)屬性,我們可以設(shè)計(jì)出下面的數(shù)據(jù)結(jié)構(gòu):

typeRequeststruct{
IDint`json:"Id"`
Timestampint`json:"Timestamp"`
StatusStatus`json:"Status"`
}

然后,假設(shè)收到的Request 的接口返回值為:

{
"Id":1234,
"Timestamp":1563362390,
"Status":0
}

到目前為止,沒(méi)有什么特殊的表達(dá),Status將會(huì)被反序列化為StatusOpen,是吧?

好的,我們來(lái)看一個(gè)未設(shè)置status返回值的請(qǐng)求(不管是出于什么原因吧)。

{
"Id":1234,
"Timestamp":1563362390
}

在這個(gè)例子中,Request結(jié)構(gòu)體的Status字段將會(huì)被初始化為默認(rèn)零值zeroed value, 對(duì)于 uint32 類型來(lái)說(shuō),值就是0。因此,StatusOpen就替換掉了原本值應(yīng)該是StatusUnknown

對(duì)于這類場(chǎng)景,把unknown value設(shè)置為枚舉類型0應(yīng)該比較合適,如下:

typeStatusuint32

const(
StatusUnknownStatus=iota
StatusOpen
StatusClose
)

這樣,即時(shí)返回的 JSON 請(qǐng)求中沒(méi)有Status屬性,結(jié)構(gòu)體RequestStatus屬性也會(huì)按我們預(yù)期的,被初始化為StatusUnknown。

性能測(cè)試

正確地進(jìn)行性能測(cè)試很困難,因?yàn)檫^(guò)程中有太多的因素會(huì)影響測(cè)試結(jié)果了。

其中一個(gè)最常見(jiàn)的錯(cuò)誤就是被一些編譯器優(yōu)化參數(shù)糊弄,讓我們以teivah/bitvector庫(kù)中的一個(gè)真實(shí)案例來(lái)進(jìn)行闡述:

funcclear(nuint64,i,juint8)uint64{
return(math.MaxUint64<1<1))&n
}

這個(gè)函數(shù)會(huì)清理給定長(zhǎng)度n的二進(jìn)制位,對(duì)這個(gè)函數(shù)進(jìn)行性能測(cè)試的話,我們可能會(huì)寫出下面的代碼:

funcBenchmarkWrong(b*testing.B){
fori:=0;i1221892080809121,10,63)
}
}

在這個(gè)性能測(cè)試中,編譯器發(fā)現(xiàn)clear函數(shù)是并沒(méi)有調(diào)用其他函數(shù),因此編譯器就會(huì)進(jìn)行inline處理。除此之外,編譯器還發(fā)現(xiàn)這個(gè)函數(shù)中也沒(méi)有side-effects。因此,clear就會(huì)被刪除,不去計(jì)算它的耗時(shí),因此這就會(huì)導(dǎo)致測(cè)試結(jié)果的不準(zhǔn)確。

一個(gè)建議是設(shè)置全局變量,如下:

varresultuint64

funcBenchmarkCorrect(b*testing.B){
varruint64
fori:=0;i1221892080809121,10,63)
}
result=r
}

這樣的話,編譯器就不知道clear函數(shù)是否會(huì)造成side-effect了,因此,性能測(cè)試的結(jié)果就會(huì)變得更加準(zhǔn)確。

拓展閱讀

指針,到處都是指針!

值傳遞的時(shí)候,會(huì)創(chuàng)建一個(gè)同值變量;而指針傳遞的時(shí)候,只是將變量地址進(jìn)行拷貝。

因此,指針傳遞總是會(huì)很快,是不?

如果你覺(jué)得是這樣,可以看一下這個(gè)例子。在這個(gè)性能測(cè)試中,一個(gè)大小為0.3K的數(shù)據(jù)結(jié)構(gòu)分別以值傳遞和指針傳遞進(jìn)行測(cè)試。0.3K 不大,但是也不能和大部分我們?nèi)粘S玫降膱?chǎng)景中的數(shù)據(jù)結(jié)構(gòu)大小相差甚遠(yuǎn),接近即可。

當(dāng)我在自己的本地環(huán)境中執(zhí)行這個(gè)性能測(cè)試代碼的時(shí)候,值傳遞比指針傳遞快了4 倍還多,是不是感覺(jué)有悖常理?

關(guān)于這個(gè)現(xiàn)象的解釋涉及到了 Go 中的內(nèi)存管理,我沒(méi)法解釋得像 William Kennedy 解釋的那樣精煉,一起來(lái)整理總結(jié)下吧:

變量可以被分配到heapstack上,粗略解釋為:

  • 棧包含哪些分配給了goroutine的隨時(shí)消失的變量,一旦函數(shù)返回,變量就會(huì)從棧中彈出
  • 堆包含共享變量,比如全局變量等

一起通過(guò)一個(gè)簡(jiǎn)單的例子來(lái)測(cè)試下:

funcgetFooValue()foo{
varresultfoo
//Dosomething
returnresult
}

result被當(dāng)前 goroutine 創(chuàng)建,這個(gè)變量就會(huì)被壓入當(dāng)前運(yùn)行棧。一旦函數(shù)返回,調(diào)用方就會(huì)收到與此變量的一份拷貝,二者值相同,但是變量地址不同。變量本身會(huì)被彈出,此時(shí)變量并不會(huì)被立即銷毀,直到它的內(nèi)存地址被另一個(gè)變量覆蓋或者被擦除,這個(gè)時(shí)候它才是真的再也不會(huì)被訪問(wèn)到了。

與此相對(duì),看一個(gè)一個(gè)指針傳遞的例子:

funcgetFooPointer()*foo{
varresultfoo
//Dosomething
return&result
}

result依舊是被當(dāng)前goroutine所創(chuàng)建,但是調(diào)用方收到的會(huì)是一個(gè)指針(指向變量的內(nèi)存地址)。如果result被棧彈出,那么調(diào)用方不可能訪問(wèn)到此變量。

在這個(gè)場(chǎng)景下,GO 的編譯器會(huì)把result放置到可以被共享的變量空間:heap。

下面來(lái)看另一個(gè)場(chǎng)景,比如:

funcmain(){
p:=&foo{}
f(p)
}

f的調(diào)用方與f所屬為同一個(gè)goroutine,變量p不會(huì)被轉(zhuǎn)換,它只是被簡(jiǎn)單放回到棧中,因此子函數(shù)依舊可以訪問(wèn)到。

舉例來(lái)說(shuō),io.Reader中的Read方法接收指針,而不是返回一個(gè),因?yàn)榉祷匾粋€(gè)切片就會(huì)被轉(zhuǎn)換到堆中。

為什么棧會(huì)這么快?這里有兩個(gè)主要的原因:

  • 棧不需要垃圾收集。正如我們所說(shuō),一個(gè)變量創(chuàng)建時(shí)被壓入棧,函數(shù)返回時(shí)從棧中彈出。根本不需要復(fù)雜的處理來(lái)回收未使用的變量。
  • 一個(gè)棧隸屬于一個(gè) goroutine,與堆中變量相比,不需要同步處理,這同樣會(huì)使得棧很快。

總結(jié)一下,當(dāng)我們創(chuàng)建一個(gè)函數(shù)的時(shí)候,我們應(yīng)該使用值傳遞而不是指針傳遞。只有我們期待某個(gè)變量被共享使用時(shí),才使用指針傳遞適用。

當(dāng)我們下次遇到性能優(yōu)化的問(wèn)題時(shí),一個(gè)可能的優(yōu)化方向就是檢查在某些場(chǎng)景下,指針傳遞是否真的會(huì)有所幫助。一個(gè)需要了解的常識(shí)是:當(dāng)使用go build -gcflags "-m -m"時(shí),編譯器會(huì)默認(rèn)將一個(gè)變量轉(zhuǎn)換到堆中。

再?gòu)?qiáng)調(diào)下,在日常開(kāi)發(fā)中,應(yīng)該總是首先考慮值傳遞。

拓展閱讀 Language Mechanics On Stacks And Pointers

干掉for/switch或者for/select

如果f函數(shù)返回了 true,會(huì)發(fā)生什么?

for{
switchf(){
casetrue:
break
casefalse:
//dosomething
}
}

break語(yǔ)句會(huì)被調(diào)用,這會(huì)導(dǎo)致switch語(yǔ)句退出,而不是 loop 退出。再看一個(gè)類似問(wèn)題:

for{
select{
case<-ch:
????????//dosomething
case<-ctx.Done():
????????break
}
}

break同樣只是退出select語(yǔ)句,而不是 for 循環(huán)。

一個(gè)可能的解決方案是使用labeled break標(biāo)簽,例如:

loop:
for{
select{
case<-ch:
????????????//dosomething
case<-ctx.Done():
????????????breakloop
}
}

錯(cuò)誤管理

Go 中的錯(cuò)誤處理機(jī)制還是有點(diǎn)簡(jiǎn)單,或許到了 Go2.0,它會(huì)變得好一點(diǎn)。

當(dāng)前標(biāo)準(zhǔn)庫(kù)只提供創(chuàng)建錯(cuò)誤類型數(shù)據(jù)結(jié)構(gòu)的方法,具體可查看 pkg/errors。

這個(gè)庫(kù)很好的展示了一些本該被遵守卻經(jīng)常不被遵守的規(guī)則的好例子。

一個(gè)錯(cuò)誤只應(yīng)該被處理一次。把錯(cuò)誤打印到日志中也是在處理錯(cuò)誤。所以一個(gè)錯(cuò)誤要么被打日志,要么被傳到調(diào)用方。

當(dāng)前的標(biāo)準(zhǔn)庫(kù),如果我們想分層化或者在錯(cuò)誤中添加上下文信息是非常困難的。接下來(lái),我們一起看個(gè)期待使用 REST 形式調(diào)用而導(dǎo)致 DB 出問(wèn)題的例子:

unabletoserveHTTPPOSTrequestforcustomer1234
|_unabletoinsertcustomercontractabcd
|_unabletocommittransaction

如果我們使用pkg/errors庫(kù),我們可能會(huì)這么做:

funcpostHandler(customerCustomer)Status{
err:=insert(customer.Contract)
iferr!=nil{
log.WithError(err).Errorf("unabletoserveHTTPPOSTrequestforcustomer%s",customer.ID)
returnStatus{ok:false}
}
returnStatus{ok:true}
}

funcinsert(contractContract)error{
err:=dbQuery(contract)
iferr!=nil{
returnerrors.Wrapf(err,"unabletoinsertcustomercontract%s",contract.ID)
}
returnnil
}

funcdbQuery(contractContract)error{
//Dosomethingthenfail
returnerrors.New("unabletocommittransaction")
}

需要我們使用errors.New來(lái)初始化錯(cuò)誤信息(如果內(nèi)部方法調(diào)用沒(méi)有返回 error 的話)。中間調(diào)用層insert, 僅僅是通過(guò)添加更多上下文信息來(lái)包裝了錯(cuò)誤。然后insert的調(diào)用方通過(guò)日志進(jìn)行了打印,每一層要么返回錯(cuò)誤,要么處理錯(cuò)誤。

有些時(shí)候,我們可能會(huì)檢查錯(cuò)誤以便于做重試處理。假如我們有一個(gè)叫db的處理數(shù)據(jù)庫(kù)的外部的包,這個(gè)庫(kù)可能會(huì)返回db.DBError 這種臨時(shí)錯(cuò)誤。到底要不要做重試處理,就看錯(cuò)誤是不是符合預(yù)期, 比如處理代碼:

funcpostHandler(customerCustomer)Status{
err:=insert(customer.Contract)
iferr!=nil{
switcherrors.Cause(err).(type){
default:
log.WithError(err).Errorf("unabletoserveHTTPPOSTrequestforcustomer%s",customer.ID)
returnStatus{ok:false}
case*db.DBError:
returnretry(customer)
}

}
returnStatus{ok:true}
}

funcinsert(contractContract)error{
err:=db.dbQuery(contract)
iferr!=nil{
returnerrors.Wrapf(err,"unabletoinsertcustomercontract%s",contract.ID)
}
returnnil
}

借助pkg/errors中的errors.Cause,便可以進(jìn)行實(shí)現(xiàn)。

一個(gè)常見(jiàn)的錯(cuò)誤就是獨(dú)立使用pkg/errors,比如:

switcherr.(type){
default:
log.WithError(err).Errorf("unabletoserveHTTPPOSTrequestforcustomer%s",customer.ID)
returnStatus{ok:false}
case*db.DBError:
returnretry(customer)
}

上面例子中,如果db.DBError被包裝了,那么重試機(jī)制將永遠(yuǎn)不會(huì)觸發(fā)。

切片初始化

有時(shí)候我們知道切片的最終長(zhǎng)度,比如:將切片Foo轉(zhuǎn)換成切片Bar,這意味著兩個(gè)切片的長(zhǎng)度會(huì)是一致的。

我經(jīng)常見(jiàn)到有人這么初始化切片:

varbar[]Bar

bars:=make([]Bar,0)

切片不是魔術(shù)結(jié)構(gòu),實(shí)際上當(dāng)空間不足時(shí),Go來(lái)動(dòng)態(tài)的維護(hù)切片的長(zhǎng)度。在這個(gè)場(chǎng)景下,一個(gè)新的更大容量的數(shù)組會(huì)自動(dòng)被創(chuàng)建,然后將舊的數(shù)組元素一個(gè)個(gè)的拷貝到新數(shù)組中。

現(xiàn)在,假設(shè)我們要多次數(shù)以千計(jì)的增加[]Foo,插入的時(shí)間復(fù)雜度可不是O(1),畢竟內(nèi)部重復(fù)了多次拷貝。

因此,如果我們知道切片最終長(zhǎng)度的話,可以采用以下策略:

  • 使用預(yù)定義長(zhǎng)度
funcconvert(foos[]Foo)[]Bar{
bars:=make([]Bar,len(foos))
fori,foo:=rangefoos{
bars[i]=fooToBar(foo)
}
returnbars
}
  • 使用 0 長(zhǎng)度,并且給一個(gè)預(yù)定義容量
funcconvert(foos[]Foo)[]Bar{
bars:=make([]Bar,0,len(foos))
for_,foo:=rangefoos{
bars=append(bars,fooToBar(foo))
}
returnbars
}

那么,這倆方法哪個(gè)更好呢?

第一個(gè)更快一點(diǎn)點(diǎn),而第二個(gè)更符合編碼預(yù)期:不考慮初始長(zhǎng)度,每次只通過(guò)append往尾部追加數(shù)據(jù)。

上下文管理

context.Context經(jīng)常被開(kāi)發(fā)者所誤解,下面看下官方的解釋:

上下文以 API 邊界形式,可攜帶截止時(shí)間、取消信號(hào)以及其他值。

這段描述通常讓人疑惑這玩意兒有啥用,咋用啊?

我們舉幾個(gè)例子,看看它到底能攜帶什么數(shù)據(jù):

  • 截止日期不管是遇到250 ms還是遇到2019-01-08 0100格式的時(shí)間,必須立刻終止執(zhí)行(執(zhí)行的內(nèi)容可能是 I/O 請(qǐng)求,等待 channel 輸入等)
  • 取消信號(hào)類似于上面,一旦接收到信號(hào),就需要立刻終止執(zhí)行后續(xù)處理。例如:接收兩個(gè)請(qǐng)求,一個(gè)是插入數(shù)據(jù),另一個(gè)是取消第一個(gè)的插入,這個(gè)場(chǎng)景就可以借助在第一個(gè)請(qǐng)求中加入一個(gè)可取消的上下文來(lái)實(shí)現(xiàn)。
  • 其他值Key-Value形式,即便都是 interface{}類型。

context 是可組合的,因此可以添加截止時(shí)間和其他 key-value 類型數(shù)據(jù);另外,多個(gè)協(xié)程可共享同一個(gè)上下文,因此取消信號(hào)可以阻止多個(gè)執(zhí)行流程。

回到正題,繼續(xù)來(lái)說(shuō)說(shuō)錯(cuò)誤問(wèn)題。

一個(gè) 基于 urface/cli (一個(gè)用于制作命令行應(yīng)用的庫(kù))Go 應(yīng)用,一旦啟動(dòng),開(kāi)發(fā)者繼承了一串上下文,使用 context 的終止信號(hào)來(lái)終止所有的執(zhí)行。當(dāng)我意識(shí)到請(qǐng)求一個(gè) gRPC 終端的時(shí)候,context 只是直接被傳遞了下去。這不是我想看到的。

相反,我們想讓 gRPC 庫(kù)在收到終止信號(hào)或者超過(guò) 100ms 處理時(shí)間時(shí)進(jìn)行取消處理。為了達(dá)到這個(gè)目標(biāo),我們可以創(chuàng)建一個(gè)簡(jiǎn)單的組合上下文,如果parent是應(yīng)用上下文的名字(通過(guò) urfave/cli 創(chuàng)建),然后我們就可以寫出下面的代碼:

ctx,cancel:=context.WithTimeout(parent,100*time.Millisecond)
response,err:=grpcClient.Send(ctx,request)

上下文不難理解,而且在我眼中,它是Go 語(yǔ)言中最棒的特色之一。

不要使用-race選項(xiàng)

我經(jīng)常見(jiàn)的一個(gè)錯(cuò)誤就是在測(cè)試時(shí)使用-race選項(xiàng)。

“即使 Go 是被設(shè)計(jì)成讓并發(fā)更容易,更少錯(cuò)誤的語(yǔ)言”, 我們?nèi)匀唤?jīng)受著很多并發(fā)問(wèn)題的折磨。

顯而易見(jiàn)的是,Go 語(yǔ)言中的 race 探查器對(duì)獨(dú)立的并發(fā)問(wèn)題而言并無(wú)幫助。不過(guò),當(dāng)測(cè)試我們的應(yīng)用時(shí)開(kāi)啟它也是很有價(jià)值的。

使用文件名作為輸入

另一個(gè)常見(jiàn)問(wèn)題就是把文件名作為函數(shù)的參數(shù)。加入我們要實(shí)現(xiàn)一個(gè)統(tǒng)計(jì)文件中空行數(shù)量的函數(shù),最自然的實(shí)現(xiàn)方式可能就是這樣的:

funccount(filenamestring)(int,error){
file,err:=os.Open(filename)
iferr!=nil{
return0,errors.Wrapf(err,"unabletoopen%s",filename)
}
deferfile.Close()

scanner:=bufio.NewScanner(file)
count:=0
forscanner.Scan(){
ifscanner.Text()==""{
count++
}
}
returncount,nil
}

filename作為函數(shù)輸入,然后我們打開(kāi)文件,再實(shí)現(xiàn)后續(xù)的邏輯,對(duì)不?

接下來(lái),在此函數(shù)的基礎(chǔ)上寫單測(cè),測(cè)試使用的變量分別代表:常規(guī)文件,空文件,使用不同編碼的文件等等。很快它就會(huì)變得難以管理。

同樣,當(dāng)我們想以同樣的邏輯來(lái)處理 HTTP 響應(yīng)體,我們就不得不重新寫一個(gè)新函數(shù)了,因?yàn)檫@個(gè)函數(shù)只接受文件名。

GO 語(yǔ)言中有兩個(gè)很棒的抽象:io.Readerio.Writer。與直接傳遞文件名不同的是,我們可以簡(jiǎn)單的傳入一個(gè)io.Reader來(lái)抽象化數(shù)據(jù)源。

它是文件還是 HTTP 的響應(yīng)體,或者是一個(gè)字節(jié)緩沖區(qū)?都不重要了,我們只需要使用Read方法就都可以搞定。在下面的例子中,我們甚至可以一行一行地讀入數(shù)據(jù)。

funccount(reader*bufio.Reader)(int,error){
count:=0
for{
line,_,err:=reader.ReadLine()
iferr!=nil{
switcherr{
default:
return0,errors.Wrapf(err,"unabletoread")
caseio.EOF:
returncount,nil
}
}
iflen(line)==0{
count++
}
}
}

打開(kāi)一個(gè)文件的職責(zé)交給count的調(diào)用方去代理就好了,如下:

file,err:=os.Open(filename)
iferr!=nil{
returnerrors.Wrapf(err,"unabletoopen%s",filename)
}
deferfile.Close()
count,err:=count(bufio.NewReader(file))

在第二種的實(shí)現(xiàn)中,數(shù)據(jù)源已經(jīng)不重要了,并且單測(cè)也可以很方便的進(jìn)行編寫,比如使用字符串來(lái)創(chuàng)建一個(gè)bufio.Reader作為數(shù)據(jù)源:

count,err:=count(bufio.NewReader(strings.NewReader("input")))

協(xié)程與循環(huán)變量

最后一個(gè)常見(jiàn)的錯(cuò)誤就是在循環(huán)結(jié)構(gòu)中使用協(xié)程。

下面例子中的輸出是什么?

ints:=[]int{1,2,3}
for_,i:=rangeints{
gofunc(){
fmt.Println("%v
",i)
}()
}

你是不是以為會(huì)是按順序輸出1 2 3?并不是哦。在這個(gè)例子中,每一個(gè)協(xié)程都會(huì)共享同一個(gè)變量實(shí)例,因此它最終大概率會(huì)輸出3 3 3

有兩種解決方案來(lái)解決類似問(wèn)題,第一個(gè)就是把循環(huán)遍歷當(dāng)做參數(shù)傳給閉包,比如:

ints:=[]int{1,2,3}
for_,i:=rangeints{
gofunc(iint){
fmt.Printf("%v
",i)
}(i)
}

另一種方式就是在循環(huán)內(nèi)部的作用域中創(chuàng)建臨時(shí)變量,比如:

ints:=[]int{1,2,3}
for_,i:=rangeints{
i:=i
gofunc(){
fmt.Printf("%v
",i)
}()
}

雖然看著i := i很奇怪,但是它真的有效。一個(gè)循環(huán)內(nèi)部意味著在另一個(gè)作用域中,因此i := i就創(chuàng)建了一個(gè)新的變量實(shí)例,稱之為i。當(dāng)然,為了可讀性我們也可以定義成一個(gè)別的名字。

轉(zhuǎn)自:

guoruibiao.blog.csdn.net/article/details/108054295

編輯:jq
聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • HTTP
    +關(guān)注

    關(guān)注

    0

    文章

    537

    瀏覽量

    35350
  • 數(shù)據(jù)源
    +關(guān)注

    關(guān)注

    1

    文章

    66

    瀏覽量

    10087
  • go語(yǔ)言
    +關(guān)注

    關(guān)注

    1

    文章

    159

    瀏覽量

    9778

原文標(biāo)題:Go 項(xiàng)目中常見(jiàn)的 10 種錯(cuò)誤

文章出處:【微信號(hào):magedu-Linux,微信公眾號(hào):馬哥Linux運(yùn)維】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏
加入交流群
微信小助手二維碼

掃碼添加小助手

加入工程師交流群

    評(píng)論

    相關(guān)推薦
    熱點(diǎn)推薦

    手機(jī)主板散熱導(dǎo)熱膠薄層涂布最佳實(shí)踐 |鉻銳特實(shí)業(yè)

    鉻銳特實(shí)業(yè)|東莞廠家|詳解手機(jī)主板導(dǎo)熱膠薄層涂布最佳實(shí)踐:推薦80-150μm厚度范圍,熱阻可降低40-50%,芯片溫度下降5-10℃。掌握精準(zhǔn)點(diǎn)膠、壓力組裝與材料選擇,實(shí)現(xiàn)高效散熱與性能
    的頭像 發(fā)表于 03-02 01:54 ?44次閱讀
    手機(jī)主板散熱導(dǎo)熱膠薄層涂布最佳<b class='flag-5'>實(shí)踐</b> |鉻銳特實(shí)業(yè)

    Go 語(yǔ)言高并發(fā)服務(wù)設(shè)計(jì)與性能調(diào)優(yōu)實(shí)戰(zhàn):從萬(wàn)級(jí)到百萬(wàn)級(jí)并發(fā)的演進(jìn)之路

    在2026年的今天,Go 語(yǔ)言已成為高并發(fā)后端服務(wù)的首選語(yǔ)言。根據(jù) Stack Overflow 最新開(kāi)發(fā)者調(diào)查: 指標(biāo) 數(shù)據(jù) Go 語(yǔ)言
    發(fā)表于 02-18 19:19

    內(nèi)核配置項(xiàng)引發(fā)網(wǎng)絡(luò)性能下降的深度剖析

    、CONFIG_PREEMPT_TRACER、CONFIG_SCHED_TRACER )的啟用,竟導(dǎo)致網(wǎng)絡(luò)性能下降10% ,關(guān)閉后借助 iperf3 測(cè)試丟包問(wèn)題消失。本文將深入剖
    的頭像 發(fā)表于 02-01 16:48 ?1664次閱讀
    內(nèi)核配置項(xiàng)引發(fā)網(wǎng)絡(luò)<b class='flag-5'>性能</b><b class='flag-5'>下降</b>的深度剖析

    低成本TLI4971/TLE4971電流傳感器評(píng)估套件——MS2Go與S2Go

    低成本TLI4971/TLE4971電流傳感器評(píng)估套件——MS2Go與S2Go 在電子工程師的日常工作,電流傳感器的評(píng)估和應(yīng)用是一個(gè)重要的環(huán)節(jié)。今天我們要介紹的是英飛凌(Infineon
    的頭像 發(fā)表于 12-19 16:50 ?830次閱讀

    單片機(jī)開(kāi)發(fā)功能安全編譯器

    ”的代碼路徑。高級(jí)語(yǔ)言,特別是C和C ++,包含數(shù)量眾多的功能,這些功能的行為不是代碼所遵循的語(yǔ)言規(guī)范所規(guī)定的。這種不確定的行為可能導(dǎo)致意外的結(jié)果和潛在的災(zāi)難性后果,而這在功能安全的應(yīng)
    發(fā)表于 12-01 06:44

    為什么ADA4530-1運(yùn)放總是?

    這個(gè)運(yùn)放的時(shí)候沒(méi)注意到GRD是做保護(hù)環(huán)用的,所以直接接了地,但是這應(yīng)該只會(huì)導(dǎo)致沒(méi)有屏蔽漏電流的效果,不會(huì)道址運(yùn)放總是吧,不知道是什么原因,我用這個(gè)運(yùn)放的時(shí)候是處在一個(gè)激光周圍,因?yàn)槲乙鸭す獯蛟?/div>
    發(fā)表于 11-28 16:15

    如何預(yù)防射頻模塊的性能下降?

    預(yù)防射頻模塊(用于干擾發(fā)生類儀器,如射頻信號(hào)發(fā)生器)性能下降,需圍繞其核心失效誘因(散熱不良、環(huán)境侵蝕、操作不當(dāng)、部件老化、負(fù)載異常),從 “環(huán)境控制、規(guī)范操作、定期維護(hù)、硬件保護(hù)、校準(zhǔn)溯源” 五大維度建立全生命周期預(yù)防體系,延緩部件老化、避免不可逆損傷。
    的頭像 發(fā)表于 10-18 10:46 ?844次閱讀

    國(guó)巨電容出現(xiàn)漏液現(xiàn)象,可能是哪些原因導(dǎo)致的?

    焊接不佳,或絕緣子與外殼、引線焊接不佳,都可能導(dǎo)致密封性能下降,從而引發(fā)漏液。 密封材料老化 :長(zhǎng)期使用后,密封材料(如橡膠塞)可能發(fā)生硬化、龜裂,失去密封性,
    的頭像 發(fā)表于 09-29 14:21 ?609次閱讀
    國(guó)巨電容出現(xiàn)漏液現(xiàn)象,<b class='flag-5'>可能</b>是哪些原因<b class='flag-5'>導(dǎo)致</b>的?

    數(shù)明半導(dǎo)體SiLM27531H柵極驅(qū)動(dòng)器在碳化硅器件的應(yīng)用

    碳化硅 MOSFET 憑借顯著的開(kāi)關(guān)性能優(yōu)勢(shì),在許多大功率應(yīng)用得到青睞。然而它的特性要求柵極驅(qū)動(dòng)電路有較高要求,以優(yōu)化碳化硅器件的開(kāi)關(guān)性能。盡管碳化硅 MOSFET 并非難以驅(qū)動(dòng),但許多常見(jiàn)的驅(qū)動(dòng)器
    的頭像 發(fā)表于 09-03 17:54 ?4641次閱讀
    數(shù)明半導(dǎo)體SiLM27531H柵極驅(qū)動(dòng)器在碳化硅器件<b class='flag-5'>中</b>的應(yīng)用

    CAN總線電容過(guò)大?三解決方案來(lái)了

    在新能源汽車路試,CAN總線傳輸異常是一個(gè)常見(jiàn)問(wèn)題。本期我們將探討由于總線電容過(guò)大導(dǎo)致下降沿過(guò)緩問(wèn)題,并介紹三有效的解決方案。CAN總線下降
    的頭像 發(fā)表于 07-22 11:36 ?701次閱讀
    CAN總線電容過(guò)大?三<b class='flag-5'>種</b>解決方案來(lái)了

    鴻蒙5開(kāi)發(fā)寶藏案例分享---性能優(yōu)化案例解析

    鴻蒙性能優(yōu)化寶藏指南:實(shí)戰(zhàn)工具與代碼案例解析 大家好呀!今天在翻鴻蒙開(kāi)發(fā)者文檔時(shí),意外挖到一個(gè) 性能優(yōu)化寶藏庫(kù) ——原來(lái)官方早就提供了超多實(shí)用工具和案例,但很多小伙伴可能沒(méi)發(fā)現(xiàn)!這篇就帶大家手把手
    發(fā)表于 06-12 16:36

    從 Java 到 Go:面向?qū)ο蟮木奕伺c云原生的輕騎兵

    Go 語(yǔ)言在 2009 年被 Google 推出,在創(chuàng)建之初便明確提出了“少即是多(Less is more)”的設(shè)計(jì)原則,強(qiáng)調(diào)“以工程效率為核心,用極簡(jiǎn)規(guī)則解決復(fù)雜問(wèn)題”。它與 Java 語(yǔ)言生態(tài)
    的頭像 發(fā)表于 04-25 11:13 ?644次閱讀

    詳解錫膏工藝的虛焊現(xiàn)象

    在錫膏工藝,虛焊(Cold Solder Joint)是一常見(jiàn)的焊接缺陷,表現(xiàn)為焊點(diǎn)表面看似連接,但實(shí)際存在電氣接觸不良或機(jī)械強(qiáng)度不足的問(wèn)題。虛焊可能導(dǎo)致產(chǎn)品功能失效、可靠性
    的頭像 發(fā)表于 04-25 09:09 ?2364次閱讀
    詳解錫膏工藝<b class='flag-5'>中</b>的虛焊現(xiàn)象

    如何成為一名合格的KaihongOS北向應(yīng)用開(kāi)發(fā)工程師

    基礎(chǔ)知識(shí) 編程語(yǔ)言:學(xué)習(xí)至少一編程語(yǔ)言,如 JavaScript和TypeScript,這些語(yǔ)言是北向應(yīng)用開(kāi)發(fā)必備的基礎(chǔ)
    發(fā)表于 04-23 06:46

    三一挖掘機(jī)一鍵啟動(dòng)開(kāi)關(guān)易的原因及更換注意事項(xiàng)

    三一挖掘機(jī)一鍵啟動(dòng)開(kāi)關(guān)易的原因雖然三一挖掘機(jī)的一鍵啟動(dòng)系統(tǒng)設(shè)計(jì)旨在提高便利性和安全性,但在實(shí)際使用,可能會(huì)出現(xiàn)一些問(wèn)題導(dǎo)致開(kāi)關(guān)易。這些
    發(fā)表于 03-12 09:29