作者:andruzhang,騰訊 IEG 后臺(tái)開(kāi)發(fā)工程師
創(chuàng)新互聯(lián)建站,是成都地區(qū)的互聯(lián)網(wǎng)解決方案提供商,用心服務(wù)為企業(yè)提供網(wǎng)站建設(shè)、重慶APP開(kāi)發(fā)、微信小程序開(kāi)發(fā)、系統(tǒng)定制網(wǎng)站建設(shè)和微信代運(yùn)營(yíng)服務(wù)。經(jīng)過(guò)數(shù)10多年的沉淀與積累,沉淀的是技術(shù)和服務(wù),讓客戶(hù)少走彎路,踏實(shí)做事,誠(chéng)實(shí)做人,用情服務(wù),致力做一個(gè)負(fù)責(zé)任、受尊敬的企業(yè)。對(duì)客戶(hù)負(fù)責(zé),就是對(duì)自己負(fù)責(zé),對(duì)企業(yè)負(fù)責(zé)。
在后臺(tái)開(kāi)發(fā)中,針對(duì)錯(cuò)誤處理,有三個(gè)維度的問(wèn)題需要解決:
一個(gè)面向過(guò)程的函數(shù),在不同的處理過(guò)程中需要 handle 不同的錯(cuò)誤信息;一個(gè)面向?qū)ο蟮暮瘮?shù),針對(duì)一個(gè)操作所返回的不同類(lèi)型的錯(cuò)誤,有可能需要進(jìn)行不同的處理。此外,在遇到錯(cuò)誤時(shí),也可以使用斷言的方式,快速中止函數(shù)流程,大大提高代碼的可讀性。
在許多高級(jí)語(yǔ)言中都提供了 try ... catch 的語(yǔ)法,函數(shù)內(nèi)部可以通過(guò)這種方案,實(shí)現(xiàn)一個(gè)統(tǒng)一的錯(cuò)誤處理邏輯。而即便是 C 這種 “中級(jí)語(yǔ)言” 雖然沒(méi)有,但是程序員也可以使用宏定義的方式,來(lái)實(shí)現(xiàn)某種程度上的錯(cuò)誤斷言。
但是,對(duì)于 Go 的情況就比較尷尬了。
我們先來(lái)看斷言,我們的目的是,僅使用一行代碼就能夠檢查錯(cuò)誤并終止當(dāng)前函數(shù)。由于沒(méi)有 throw,沒(méi)有宏,如果要實(shí)現(xiàn)一行斷言,有兩種方法。
第一種是把 if 的錯(cuò)誤判斷寫(xiě)在一行內(nèi),比如:
第二種方法是借用 panic 函數(shù),結(jié)合 recover 來(lái)實(shí)現(xiàn):
這兩種方法都值得商榷。
首先,將 if 寫(xiě)在同一行內(nèi)的問(wèn)題有:
至于第二種方法,我們要分情況看;
不過(guò)使用 panic 來(lái)斷言的方案,雖然在業(yè)務(wù)邏輯中基本上不用,但在測(cè)試場(chǎng)景下則是非常常見(jiàn)的。測(cè)試嘛,用牛刀有何不可?稍微大一點(diǎn)的系統(tǒng)開(kāi)銷(xiāo)也沒(méi)啥問(wèn)題。對(duì)于 Go 來(lái)說(shuō),非常熱門(mén)的單元測(cè)試框架 goconvey 就是使用 panic 機(jī)制來(lái)實(shí)現(xiàn)單元測(cè)試中的斷言,用的人都說(shuō)好。
綜上,在 Go 中,對(duì)于業(yè)務(wù)代碼,筆者不建議采用斷言,遇到錯(cuò)誤的時(shí)候建議還是老老實(shí)實(shí)采用這種格式:
而在單測(cè)代碼中,則完全可以大大方方地采用類(lèi)似于 goconvey 之類(lèi)基于 panic 機(jī)制的斷言。
眾所周知 Go 是沒(méi)有 try ... catch 的,而且從官方的態(tài)度來(lái)看,短時(shí)間內(nèi)也沒(méi)有考慮的計(jì)劃。但程序員有這個(gè)需求呀。筆者采用的方法,是將需要返回的 err 變量在函數(shù)內(nèi)部全局化,然后結(jié)合 defer 統(tǒng)一處理:
這種方案要特別注意變量作用域問(wèn)題.比如前面的 if err = DoSomething(); err != nil { 行,如果我們將 err = ... 改為 err := ...,那么這一行中的 err 變量和函數(shù)最前面定義的 (err error) 不是同一個(gè)變量,因此即便在此處發(fā)生了錯(cuò)誤,但是在 defer 函數(shù)中無(wú)法捕獲到 err 變量了。
在 try ... catch 方面,筆者其實(shí)沒(méi)有特別好的方法來(lái)模擬,即便是上面的方法也有一個(gè)很讓人頭疼的問(wèn)題:defer 寫(xiě)法導(dǎo)致錯(cuò)誤處理前置,而正常邏輯后置了,從可讀性的角度來(lái)說(shuō)非常不友好。因此也希望讀者能夠指教。同時(shí)還是希望 Go 官方能夠繼續(xù)迭代,支持這種語(yǔ)法。
這一點(diǎn)在 Go 里面,一開(kāi)始看起來(lái)還是比較統(tǒng)一的,這就是 Go 最開(kāi)始就定義的 error 類(lèi)型,以系統(tǒng)標(biāo)準(zhǔn)的方式,統(tǒng)一了進(jìn)程內(nèi)函數(shù)級(jí)的錯(cuò)誤返回模式。調(diào)用方使用 if err != nil 的統(tǒng)一模式,來(lái)判斷一個(gè)調(diào)用是不是成功了。
但是隨著 Go 的逐步推廣,由于 error 接口的高自由度,程序員們對(duì)于 “如何判斷該錯(cuò)誤是什么錯(cuò)誤” 的時(shí)候,出現(xiàn)了分歧。
在 Go 1.13 之前,對(duì)于 error 類(lèi)型的傳遞,有三種常見(jiàn)的模式:
這個(gè)流派很簡(jiǎn)單,就是將各種錯(cuò)誤信息直接定義為一個(gè)類(lèi)枚舉值的模式,比如:
當(dāng)遇到相應(yīng)的錯(cuò)誤信息時(shí),直接返回對(duì)應(yīng)的 error 類(lèi)枚舉值就行了。對(duì)于調(diào)用方也非常方便,可以采用 switch - case 來(lái)判斷錯(cuò)誤類(lèi)型:
個(gè)人覺(jué)得這種設(shè)計(jì)模式本質(zhì)上還是 C error code 模式。
這種流派則是充分使用了 “error 是一個(gè) interface” 的特性,重新自定義一個(gè) error 類(lèi)型。一方面是用不同的類(lèi)型來(lái)表示不同的錯(cuò)誤分類(lèi),另一方面則能夠?qū)崿F(xiàn)對(duì)于同一錯(cuò)誤類(lèi)型,能夠給調(diào)用方提供更佳詳盡的信息。舉個(gè)例子,我們可以定義多個(gè)不同的錯(cuò)誤類(lèi)型如下:
對(duì)于調(diào)用方,則通過(guò)以下代碼來(lái)判斷不同的錯(cuò)誤:
這種模式,一方面可以透?jìng)鞯讓渝e(cuò)誤,另一方面又可以添加自定義的信息。但對(duì)于調(diào)用方而言,災(zāi)難在于如果要判斷某一個(gè)錯(cuò)誤的具體類(lèi)型,只能用 strings.Contains() 來(lái)實(shí)現(xiàn),而錯(cuò)誤的具體描述文字是不可靠的,同一類(lèi)型的信息可能會(huì)有不同的表達(dá);而在 fmt.Errorf 的過(guò)程中,各個(gè)業(yè)務(wù)添加的額外信息也可能會(huì)有不同的文字,這帶來(lái)了極大的不可靠性,提高了模塊之間的耦合度。
在 go 1.13 版本發(fā)布之后,針對(duì) fmt.Errorf 增加了 wraping 功能,并在 errors 包中添加了 Is() 和 As() 函數(shù)。關(guān)于這個(gè)模式的原理和使用已經(jīng)有很多文章了,本文就不再贅述。
這個(gè)功能,合并并改造了前文的所謂 “== 流派” 和 “fmt.Errorf” 流派,統(tǒng)一使用 errors.Is() 函數(shù);此外,也算是官方對(duì)類(lèi)型斷言流派的認(rèn)可(專(zhuān)門(mén)用 As() 函數(shù)來(lái)支持)。
在實(shí)際應(yīng)用中,函數(shù)/模塊透?jìng)麇e(cuò)誤時(shí),應(yīng)該采用 Go 的 error wrapping 模式,也就是 fmt.Errorf() 配合 %w 使用,業(yè)務(wù)方可以放心地添加自己的錯(cuò)誤信息,只要調(diào)用方統(tǒng)一采用 errors.Is() 和 errors.As() 即可。
服務(wù)/系統(tǒng)層面的錯(cuò)誤信息返回,大部分協(xié)議都可以看成是 code - message 模式或者是其變體:
這種模式的特點(diǎn)是:code 是給程序代碼使用的,代碼判斷這是一個(gè)什么類(lèi)型的錯(cuò)誤,進(jìn)入相應(yīng)的分支處理;而 message 是給人看的,程序可以以某種形式拋出或者記錄這個(gè)錯(cuò)誤信息,供用戶(hù)查看。
在這一層面有什么問(wèn)題呢?code for computer,message for user,好像挺好的。
但有時(shí)候,我們可能會(huì)收到用戶(hù)/客戶(hù)反饋一個(gè)問(wèn)題:“XXX 報(bào)錯(cuò)了,幫忙看看什么問(wèn)題?”。用戶(hù)看不懂我們的錯(cuò)誤提示嗎?
在筆者的經(jīng)驗(yàn)中,我們?cè)谑褂?code - message 機(jī)制的時(shí)候,特別是業(yè)務(wù)初期,難以避免的是前后端的設(shè)計(jì)文案沒(méi)能完整地覆蓋所有的錯(cuò)誤用例,或者是錯(cuò)誤極其罕見(jiàn)。因此當(dāng)出現(xiàn)錯(cuò)誤時(shí),提示曖昧不清(甚至是直接提示錯(cuò)誤信息),導(dǎo)致用戶(hù)從錯(cuò)誤信息中找到解決方案
在這種情況下,盡量覆蓋所有錯(cuò)誤路徑肯定是最完美的方法。不過(guò)在做到這一點(diǎn)之前,碼農(nóng)們往往有下面的解決方案:
既要隱藏信息,又要暴露信息,我可以摔盤(pán)子嗎……
這里,筆者從日益普及的短信驗(yàn)證碼有了個(gè)靈感——人的短期記憶對(duì) 4 個(gè)字符還是比較強(qiáng)的,因此我們可以考慮把錯(cuò)誤代碼縮短到 4 個(gè)字符——不區(qū)分大小寫(xiě),因?yàn)槿绻嗽谟洃洉r(shí)還要記錄大小寫(xiě)的話(huà),難度會(huì)增加不少。
怎么用 4 個(gè)字符表示盡量多的數(shù)據(jù)呢?數(shù)字+字母總共有 36 個(gè)字符,理論上使用 4 位 36 進(jìn)制可以表示 36x36x36x36 = 1679616 個(gè)值。因此我們只要找到一個(gè)針對(duì)錯(cuò)誤信息字符串的哈希算法,把輸出值限制在 1679616 范圍內(nèi)就行了。
這里我采用的是 MD5 作為例子。MD5 的輸出是 128 位,理論上我可以取 MD5 的輸出,模 1679616 就可以得到一個(gè)簡(jiǎn)易的結(jié)果。實(shí)際上為了減少除法運(yùn)算,我采用的是取高 20 位(0xFFFFF)的簡(jiǎn)易方式(20 位二進(jìn)制的最大值為 1048575),然后將這個(gè)數(shù)字轉(zhuǎn)成 36 進(jìn)制的字符串輸出。
當(dāng)出現(xiàn)異常錯(cuò)誤時(shí),我們可以將 message 的提示信息如下展示:“未知錯(cuò)誤,錯(cuò)誤代碼 30EV,如需協(xié)助,請(qǐng)聯(lián)系 XXX”。順帶一提,30EV 是 "Access denied for user 'db_user'@'127.0.0.1'" 的計(jì)算結(jié)果,這樣一來(lái),我就對(duì)調(diào)用方隱藏了敏感信息。
至于后臺(tái)側(cè),還是需要實(shí)實(shí)在在地將這個(gè)哈希值和具體的錯(cuò)誤信息記錄在日志或者其他支持搜索的渠道里。當(dāng)用戶(hù)提供該代碼時(shí),可以快速定位。
這種方案的優(yōu)點(diǎn)很明顯:
簡(jiǎn)易的錯(cuò)誤碼生成代碼如下:
當(dāng)然這種方案也有局限性,筆者能想到的是需要注意以下兩點(diǎn):
此外,筆者需要再?gòu)?qiáng)調(diào)的是:在開(kāi)發(fā)中,針對(duì)各種不同的、正式的錯(cuò)誤用例依然需要完整覆蓋,盡可能通過(guò)已有的 code - message 機(jī)制將足夠清晰的信息告知主調(diào)方。這種 hashcode 的錯(cuò)誤代碼生成方法,僅適用于錯(cuò)誤用例遺漏、或者是快速迭代過(guò)程中,用于發(fā)現(xiàn)和調(diào)試遺漏的錯(cuò)誤用例的臨時(shí)方案。
1、簡(jiǎn)單易學(xué)。
Go語(yǔ)言的作者本身就很懂C語(yǔ)言,所以同樣Go語(yǔ)言也會(huì)有C語(yǔ)言的基因,所以對(duì)于程序員來(lái)說(shuō),Go語(yǔ)言天生就會(huì)讓人很熟悉,容易上手。
2、并發(fā)性好。
Go語(yǔ)言天生支持并發(fā),可以充分利用多核,輕松地使用并發(fā)。 這是Go語(yǔ)言最大的特點(diǎn)。
描述
Go的語(yǔ)法接近C語(yǔ)言,但對(duì)于變量的聲明有所不同。Go支持垃圾回收功能。Go的并行模型是以東尼·霍爾的通信順序進(jìn)程(CSP)為基礎(chǔ),采取類(lèi)似模型的其他語(yǔ)言包括Occam和Limbo,但它也具有Pi運(yùn)算的特征,比如通道傳輸。
在1.8版本中開(kāi)放插件(Plugin)的支持,這意味著現(xiàn)在能從Go中動(dòng)態(tài)加載部分函數(shù)。
與C++相比,Go并不包括如枚舉、異常處理、繼承、泛型、斷言、虛函數(shù)等功能,但增加了 切片(Slice) 型、并發(fā)、管道、垃圾回收、接口(Interface)等特性的語(yǔ)言級(jí)支持。
可以。
1.首先,打開(kāi)鼠標(biāo)驅(qū)動(dòng)(鼠標(biāo)宏設(shè)置都是大同小異),選擇“宏”設(shè)置,然后進(jìn)入宏設(shè)置頁(yè)面。
2.點(diǎn)擊鼠標(biāo)界面上的“側(cè)面圖”進(jìn)入子鍵“宏”設(shè)置
3. ?進(jìn)入側(cè)鍵“宏”設(shè)置后,編輯界面上的"+"號(hào)進(jìn)入錄制界面
4. ?點(diǎn)擊界面上的“錄制”按鈕即可錄制編輯,?所有鼠標(biāo)帶“宏”設(shè)置功能的操作方法都是一樣的,就像編寫(xiě)程序一樣設(shè)置響應(yīng)事件即可完成你想要的操作。
5. ?根據(jù)你在游戲里需要的操作,進(jìn)行所需要的游戲編輯
《反恐精英:全球攻勢(shì)》(英文:Counter-Strike: Global Offensive),通常簡(jiǎn)稱(chēng)為CSGO。是由Valve Software開(kāi)發(fā)的射擊游戲作品,由Steam發(fā)行。
它由最新一代起源引擎(起源引擎2012)開(kāi)發(fā),并繼承了絕大部分的經(jīng)典反恐精英設(shè)計(jì),是繼《反恐精英:起源》后第五部《反恐精英》作品。它主要針對(duì)《CS1.6》和《CS:起源》對(duì)新手玩家不太友好這點(diǎn)做出改進(jìn),《CS:GO》還會(huì)通過(guò)綜合勝負(fù)百分比、作戰(zhàn)場(chǎng)次和擊殺比率等一系列因素,將實(shí)力更接近的玩家組合成對(duì)抗雙方,開(kāi)發(fā)者希望這樣可以讓比賽更加平衡。
2016年7月27日,完美世界宣布獲得該游戲在中國(guó)大陸運(yùn)營(yíng)代理權(quán),CS:GO正式登陸國(guó)服。
golang定義可變參數(shù)的函數(shù)方法是:
—- 采用ANSI標(biāo)準(zhǔn)形式時(shí),參數(shù)個(gè)數(shù)可變的函數(shù)的原型聲明是:
type funcname(type para1, type para2, …)
—- 這種形式至少需要一個(gè)普通的形式參數(shù),后面的省略號(hào)不表示省略,而是函數(shù)原型的一部分。type是函數(shù)返回值和形式參數(shù)的類(lèi)型。
—- 采用與UNIX System V兼容的聲明方式時(shí),參數(shù)個(gè)數(shù)可變的函數(shù)原型是:
type funcname(va_alist)
va_dcl
—- 這種形式不需要提供任何普通的形式參數(shù)。
type是函數(shù)返回值的類(lèi)型。va_dcl是對(duì)函數(shù)原型聲明中參數(shù)va_alist的詳細(xì)聲明,實(shí)際是一個(gè)宏定義,對(duì)不同的硬件平臺(tái)采用不同的類(lèi)型來(lái)定義,但在最后都包括了一個(gè)分號(hào)。因此va_dcl后不再需要加上分號(hào)了。va_dcl在代碼中必須原樣給出。va_alist在VC中可以原樣給出,也可以略去。
此外,采用頭文件stdarg.h編寫(xiě)的程序是符合ANSI標(biāo)準(zhǔn)的,可以在各種操作系統(tǒng)和硬件上運(yùn)行;而采用頭文件varargs.h的方式僅僅是為了與以前的程序兼容。所以建議使用前者。
很遺憾,Go中沒(méi)有這樣的設(shè)計(jì),當(dāng)然,目前大多數(shù)相對(duì)高級(jí)的語(yǔ)言都取消了宏定義的方法,雖然這樣降低了程序員對(duì)程序的掌控能力,但是這樣更容易保證程序運(yùn)行的一致性。俗話(huà)說(shuō),有舍也有得吧。
對(duì)于想要實(shí)現(xiàn)Release版本與Develop版本體現(xiàn)不一樣的運(yùn)行效果,可以通過(guò)定義特殊的標(biāo)記常量或者變量來(lái)實(shí)現(xiàn),這一點(diǎn)在Java等很多語(yǔ)言上都是一樣的。
1、學(xué)習(xí)曲線
它包含了類(lèi)C語(yǔ)法、GC內(nèi)置和工程工具。這一點(diǎn)非常重要,因?yàn)镚o語(yǔ)言容易學(xué)習(xí),所以一個(gè)普通的大學(xué)生花一個(gè)星期就能寫(xiě)出來(lái)可以上手的、高性能的應(yīng)用。在國(guó)內(nèi)大家都追求快,這也是為什么國(guó)內(nèi)Go流行的原因之一。
2、效率
Go擁有接近C的運(yùn)行效率和接近PHP的開(kāi)發(fā)效率,這就很有利的支撐了上面大家追求快速的需求。
3、出身名門(mén)、血統(tǒng)純正
之所以說(shuō)Go語(yǔ)言出身名門(mén),是因?yàn)槲覀冎繥o語(yǔ)言出自Google公司,這個(gè)公司在業(yè)界的知名度和實(shí)力自然不用多說(shuō)。Google公司聚集了一批牛人,在各種編程語(yǔ)言稱(chēng)雄爭(zhēng)霸的局面下推出新的編程語(yǔ)言,自然有它的戰(zhàn)略考慮。而且從Go語(yǔ)言的發(fā)展態(tài)勢(shì)來(lái)看,Google對(duì)它這個(gè)新的寵兒還是很看重的,Go自然有一個(gè)良好的發(fā)展前途。我們看看Go語(yǔ)言的主要?jiǎng)?chuàng)造者,血統(tǒng)純正這點(diǎn)就可見(jiàn)端倪了。
4、組合的思想、無(wú)侵入式的接口
Go語(yǔ)言可以說(shuō)是開(kāi)發(fā)效率和運(yùn)行效率二者的完美融合,天生的并發(fā)編程支持。Go語(yǔ)言支持當(dāng)前所有的編程范式,包括過(guò)程式編程、面向?qū)ο缶幊桃约昂瘮?shù)式編程。
5、強(qiáng)大的標(biāo)準(zhǔn)庫(kù)
這包括互聯(lián)網(wǎng)應(yīng)用、系統(tǒng)編程和網(wǎng)絡(luò)編程。Go里面的標(biāo)準(zhǔn)庫(kù)基本上已經(jīng)是非常穩(wěn)定,特別是我這里提到的三個(gè),網(wǎng)絡(luò)層、系統(tǒng)層的庫(kù)非常實(shí)用。
6、部署方便
我相信這一點(diǎn)是很多人選擇Go的最大理由,因?yàn)椴渴鹛奖悖袁F(xiàn)在也有很多人用Go開(kāi)發(fā)運(yùn)維程序。
7、簡(jiǎn)單的并發(fā)
它包含降低心智的并發(fā)和簡(jiǎn)易的數(shù)據(jù)同步,我覺(jué)得這是Go最大的特色。之所以寫(xiě)正確的并發(fā)、容錯(cuò)和可擴(kuò)展的程序如此之難,是因?yàn)槲覀冇昧隋e(cuò)誤的工具和錯(cuò)誤的抽象,Go可以說(shuō)這一塊做的相當(dāng)簡(jiǎn)單。
8、穩(wěn)定性
Go擁有強(qiáng)大的編譯檢查、嚴(yán)格的編碼規(guī)范和完整的軟件生命周期工具,具有很強(qiáng)的穩(wěn)定性,穩(wěn)定壓倒一切。那么為什么Go相比于其他程序會(huì)更穩(wěn)定呢?這是因?yàn)镚o提供了軟件生命周期的各個(gè)環(huán)節(jié)的工具,如go
tool、gofmt、go test。