云和安全管理服務(wù)專家新鈦云服 張春翻譯
冀州網(wǎng)站建設(shè)公司成都創(chuàng)新互聯(lián)公司,冀州網(wǎng)站設(shè)計(jì)制作,有大型網(wǎng)站制作公司豐富經(jīng)驗(yàn)。已為冀州上千提供企業(yè)網(wǎng)站建設(shè)服務(wù)。企業(yè)網(wǎng)站搭建\成都外貿(mào)網(wǎng)站建設(shè)公司要多少錢,請(qǐng)找那個(gè)售后服務(wù)好的冀州做網(wǎng)站的公司定做!
這種方法有幾個(gè)缺點(diǎn)。首先,它可以對(duì)程序員隱藏錯(cuò)誤處理路徑,特別是在捕獲異常不是強(qiáng)制性的情況下,例如在 Python 中。即使在具有必須處理的 Java 風(fēng)格的檢查異常的語(yǔ)言中,如果在與原始調(diào)用不同的級(jí)別上處理錯(cuò)誤,也并不總是很明顯錯(cuò)誤是從哪里引發(fā)的。
我們都見(jiàn)過(guò)長(zhǎng)長(zhǎng)的代碼塊包裝在一個(gè) try-catch 塊中。在這種情況下,catch 塊實(shí)際上充當(dāng) goto 語(yǔ)句,這通常被認(rèn)為是有害的(奇怪的是,C 中的關(guān)鍵字被認(rèn)為可以接受的少數(shù)用例之一是錯(cuò)誤后清理,因?yàn)樵撜Z(yǔ)言沒(méi)有 Golang- 樣式延遲語(yǔ)句)。
如果你確實(shí)從源頭捕獲異常,你會(huì)得到一個(gè)不太優(yōu)雅的 Go 錯(cuò)誤模式版本。這可能會(huì)解決混淆代碼的問(wèn)題,但會(huì)遇到另一個(gè)問(wèn)題:性能。在諸如 Java 之類的語(yǔ)言中,拋出異??赡鼙群瘮?shù)的常規(guī)返回慢數(shù)百倍。
Java 中最大的性能成本是由打印異常的堆棧跟蹤造成的,這是昂貴的,因?yàn)檫\(yùn)行的程序必須檢查編譯它的源代碼 。僅僅進(jìn)入一個(gè) try 塊也不是空閑的,因?yàn)樾枰4?CPU 內(nèi)存寄存器的先前狀態(tài),因?yàn)樗鼈兛赡苄枰趻伋霎惓5那闆r下恢復(fù)。
如果您將異常視為通常不會(huì)發(fā)生的異常情況,那么異常的缺點(diǎn)并不重要。這可能是傳統(tǒng)的單體應(yīng)用程序的情況,其中大部分代碼庫(kù)不必進(jìn)行網(wǎng)絡(luò)調(diào)用——一個(gè)操作格式良好的數(shù)據(jù)的函數(shù)不太可能遇到錯(cuò)誤(除了錯(cuò)誤的情況)。一旦您在代碼中添加 I/O,無(wú)錯(cuò)誤代碼的夢(mèng)想就會(huì)破滅:您可以忽略錯(cuò)誤,但不能假裝它們不存在!
try {
doSometing()
} catch (IOException e) {
// ignore it
}
與大多數(shù)其他編程語(yǔ)言不同,Golang 接受錯(cuò)誤是不可避免的。 如果在單體架構(gòu)時(shí)代還不是這樣,那么在今天的模塊化后端服務(wù)中,服務(wù)通常和外部 API 調(diào)用、數(shù)據(jù)庫(kù)讀取和寫(xiě)入以及與其他服務(wù)通信 。
以上所有方法都可能失敗,解析或驗(yàn)證從它們接收到的數(shù)據(jù)(通常在無(wú)模式 JSON 中)也可能失敗。Golang 使可以從這些調(diào)用返回的錯(cuò)誤顯式化,與普通返回值的等級(jí)相同。從函數(shù)調(diào)用返回多個(gè)值的能力支持這一點(diǎn),這在大多數(shù)語(yǔ)言中通常是不可能的。Golang 的錯(cuò)誤處理系統(tǒng)不僅僅是一種語(yǔ)言怪癖,它是一種將錯(cuò)誤視為替代返回值的完全不同的方式!
重復(fù) if err != nil
對(duì) Go 錯(cuò)誤處理的一個(gè)常見(jiàn)批評(píng)是被迫重復(fù)以下代碼塊:
res, err := doSomething()
if err != nil {
// Handle error
}
對(duì)于新用戶來(lái)說(shuō),這可能會(huì)覺(jué)得沒(méi)用而且浪費(fèi)行數(shù):在其他語(yǔ)言中需要 3 行的函數(shù)很可能會(huì)增長(zhǎng)到 12 行 :
這么多行代碼!這么低效!如果您認(rèn)為上述內(nèi)容不優(yōu)雅或浪費(fèi)代碼,您可能忽略了我們檢查代碼中的錯(cuò)誤的全部原因:我們需要能夠以不同的方式處理它們!對(duì) API 或數(shù)據(jù)庫(kù)的調(diào)用可能會(huì)被重試。
有時(shí)事件的順序很重要:調(diào)用外部 API 之前發(fā)生的錯(cuò)誤可能不是什么大問(wèn)題(因?yàn)閿?shù)據(jù)從未通過(guò)發(fā)送),而 API 調(diào)用和寫(xiě)入本地?cái)?shù)據(jù)庫(kù)之間的錯(cuò)誤可能需要立即注意,因?yàn)?這可能意味著系統(tǒng)最終處于不一致的狀態(tài)。即使我們只想將錯(cuò)誤傳播給調(diào)用者,我們也可能希望用失敗的解釋來(lái)包裝它們,或者為每個(gè)錯(cuò)誤返回一個(gè)自定義錯(cuò)誤類型。
并非所有錯(cuò)誤都是相同的,并且向調(diào)用者返回適當(dāng)?shù)腻e(cuò)誤是 API 設(shè)計(jì)的重要部分,無(wú)論是對(duì)于內(nèi)部包還是 REST API 。
不必?fù)?dān)心在你的代碼中重復(fù) if err != nil ——這就是 Go 中的代碼應(yīng)該看起來(lái)的樣子。
自定義錯(cuò)誤類型和錯(cuò)誤包裝
從導(dǎo)出的方法返回錯(cuò)誤時(shí),請(qǐng)考慮指定自定義錯(cuò)誤類型,而不是單獨(dú)使用錯(cuò)誤字符串。字符串在意外代碼中是可以的,但在導(dǎo)出的函數(shù)中,它們成為函數(shù)公共 API 的一部分。更改錯(cuò)誤字符串將是一項(xiàng)重大更改——如果沒(méi)有明確的錯(cuò)誤類型,需要檢查返回錯(cuò)誤類型的單元測(cè)試將不得不依賴原始字符串值!事實(shí)上,基于字符串的錯(cuò)誤也使得在私有方法中測(cè)試不同的錯(cuò)誤案例變得困難,因此您也應(yīng)該考慮在包中使用它們?;氐藉e(cuò)誤與異常的爭(zhēng)論,返回錯(cuò)誤也使代碼比拋出異常更容易測(cè)試,因?yàn)殄e(cuò)誤只是要檢查的返回值。不需要測(cè)試框架或在測(cè)試中捕獲異常 。
可以在 database/sql 包中找到簡(jiǎn)單自定義錯(cuò)誤類型的一個(gè)很好的示例。它定義了一個(gè)導(dǎo)出常量列表,表示包可以返回的錯(cuò)誤類型,最著名的是 sql.ErrNoRows。雖然從 API 設(shè)計(jì)的角度來(lái)看,這種特定的錯(cuò)誤類型有點(diǎn)問(wèn)題(您可能會(huì)爭(zhēng)辯說(shuō) API 應(yīng)該返回一個(gè)空結(jié)構(gòu)而不是錯(cuò)誤),但任何需要檢查空行的應(yīng)用程序都可以導(dǎo)入該常量并在代碼中使用它不必?fù)?dān)心錯(cuò)誤消息本身會(huì)改變和破壞代碼。
對(duì)于更復(fù)雜的錯(cuò)誤處理,您可以通過(guò)實(shí)現(xiàn)返回錯(cuò)誤字符串的 Error() 方法來(lái)定義自定義錯(cuò)誤類型。自定義錯(cuò)誤可以包括元數(shù)據(jù),例如錯(cuò)誤代碼或原始請(qǐng)求參數(shù)。如果您想表示錯(cuò)誤類別,它們很有用。DigitalOcean 的本教程展示了如何使用自定義錯(cuò)誤類型來(lái)表示可以重試的一類臨時(shí)錯(cuò)誤。
通常,錯(cuò)誤會(huì)通過(guò)將低級(jí)錯(cuò)誤與更高級(jí)別的解釋包裝起來(lái),從而在程序的調(diào)用堆棧中傳播。例如,數(shù)據(jù)庫(kù)錯(cuò)誤可能會(huì)以下列格式記錄在 API 調(diào)用處理程序中:調(diào)用 CreateUser 端點(diǎn)時(shí)出錯(cuò):查詢數(shù)據(jù)庫(kù)時(shí)出錯(cuò):pq:檢測(cè)到死鎖。這很有用,因?yàn)樗梢詭椭覀兏欏e(cuò)誤在系統(tǒng)中傳播的過(guò)程,向我們展示根本原因(數(shù)據(jù)庫(kù)事務(wù)引擎中的死鎖)以及它對(duì)更廣泛系統(tǒng)的影響(調(diào)用者無(wú)法創(chuàng)建新用戶)。
自 Go 1.13 以來(lái),此模式具有特殊的語(yǔ)言支持,并帶有錯(cuò)誤包裝。通過(guò)在創(chuàng)建字符串錯(cuò)誤時(shí)使用 %w 動(dòng)詞,可以使用 Unwrap() 方法訪問(wèn)底層錯(cuò)誤。除了比較錯(cuò)誤相等性的函數(shù) errors.Is() 和 errors.As() 外,程序還可以獲取包裝錯(cuò)誤的原始類型或標(biāo)識(shí)。這在某些情況下可能很有用,盡管我認(rèn)為在確定如何處理所述錯(cuò)誤時(shí)最好使用頂級(jí)錯(cuò)誤的類型。
Panics
不要 panic()!長(zhǎng)時(shí)間運(yùn)行的應(yīng)用程序應(yīng)該優(yōu)雅地處理錯(cuò)誤而不是panic。即使在無(wú)法恢復(fù)的情況下(例如在啟動(dòng)時(shí)驗(yàn)證配置),最好記錄一個(gè)錯(cuò)誤并優(yōu)雅地退出。panic比錯(cuò)誤消息更難診斷,并且可能會(huì)跳過(guò)被推遲的重要關(guān)閉代碼。
Logging
我還想簡(jiǎn)要介紹一下日志記錄,因?yàn)樗翘幚礤e(cuò)誤的關(guān)鍵部分。通常你能做的最好的事情就是記錄收到的錯(cuò)誤并繼續(xù)下一個(gè)請(qǐng)求。
除非您正在構(gòu)建簡(jiǎn)單的命令行工具或個(gè)人項(xiàng)目,否則您的應(yīng)用程序應(yīng)該使用結(jié)構(gòu)化的日志庫(kù),該庫(kù)可以為日志添加時(shí)間戳,并提供對(duì)日志級(jí)別的控制。最后一部分特別重要,因?yàn)樗鼘⒃试S您突出顯示應(yīng)用程序記錄的所有錯(cuò)誤和警告。通過(guò)幫助將它們與信息級(jí)日志分開(kāi),這將為您節(jié)省無(wú)數(shù)時(shí)間。
微服務(wù)架構(gòu)還應(yīng)該在日志行中包含服務(wù)的名稱以及機(jī)器實(shí)例的名稱。默認(rèn)情況下記錄這些時(shí),程序代碼不必?fù)?dān)心包含它們。您也可以在日志的結(jié)構(gòu)化部分中記錄其他字段,例如收到的錯(cuò)誤(如果您不想將其嵌入日志消息本身)或有問(wèn)題的請(qǐng)求或響應(yīng)。只需確保您的日志沒(méi)有泄露任何敏感數(shù)據(jù),例如密碼、API 密鑰或用戶的個(gè)人數(shù)據(jù)!
對(duì)于日志庫(kù),我過(guò)去使用過(guò) logrus 和 zerolog,但您也可以選擇其他結(jié)構(gòu)化日志庫(kù)。如果您想了解更多信息,互聯(lián)網(wǎng)上有許多關(guān)于如何使用這些的指南。如果您將應(yīng)用程序部署到云中,您可能需要日志庫(kù)上的適配器來(lái)根據(jù)您的云平臺(tái)的日志 API 格式化日志 - 沒(méi)有它,云平臺(tái)可能無(wú)法檢測(cè)到日志級(jí)別等某些功能。
如果您在應(yīng)用程序中使用調(diào)試級(jí)別日志(默認(rèn)情況下通常不記錄),請(qǐng)確保您的應(yīng)用程序可以輕松更改日志級(jí)別,而無(wú)需更改代碼。更改日志級(jí)別還可以暫時(shí)使信息級(jí)別甚至警告級(jí)別的日志靜音,以防它們突然變得過(guò)于嘈雜并開(kāi)始淹沒(méi)錯(cuò)誤。您可以使用在啟動(dòng)時(shí)檢查以設(shè)置日志級(jí)別的環(huán)境變量來(lái)實(shí)現(xiàn)這一點(diǎn)。
原文:
goto語(yǔ)句也稱為無(wú)條件轉(zhuǎn)移語(yǔ)句,其一般格式如下: goto 語(yǔ)句標(biāo)號(hào); 其中語(yǔ)句標(biāo)號(hào)是按標(biāo)識(shí)符規(guī)定書(shū)寫(xiě)的符號(hào), 放在某一語(yǔ)句行的前面,標(biāo)號(hào)后加冒號(hào)(:)。語(yǔ)句標(biāo)號(hào)起標(biāo)識(shí)語(yǔ)句的作用,與goto 語(yǔ)句配合使用。
如: label: i++;
loop: while(x7);
goto loop;
C語(yǔ)言不限制程序中使用標(biāo)號(hào)的次數(shù),但各標(biāo)號(hào)不得重名。goto語(yǔ)句的語(yǔ)義是改變程序流向, 轉(zhuǎn)去執(zhí)行語(yǔ)句標(biāo)號(hào)所標(biāo)識(shí)的語(yǔ)句。
goto語(yǔ)句通常與條件語(yǔ)句配合使用。可用來(lái)實(shí)現(xiàn)條件轉(zhuǎn)移, 構(gòu)成循環(huán),跳出循環(huán)體等功能。
擴(kuò)展資料:
go to在C語(yǔ)言中的應(yīng)用:
統(tǒng)計(jì)從鍵盤輸入一行字符的個(gè)數(shù)。
#includestdio.h
int?n=0;
int?main(void)?{
printf("input?a?string:?");
? loop:?if?(getchar()!='\n')?{
n++;
?goto?loop;
}
printf("output:?%d\n",n);
}
例如輸入:abcdefghijklmnopqrstuvwxyz
然后回車Enter
輸出:26
本例用if語(yǔ)句和goto語(yǔ)句構(gòu)成循環(huán)結(jié)構(gòu)。當(dāng)輸入字符不為'\n'時(shí)即執(zhí)行n++進(jìn)行計(jì)數(shù)。
然后轉(zhuǎn)移至if語(yǔ)句循環(huán)執(zhí)行,直至輸入字符為'\n'才停止循環(huán)。
參考資料:百度百科-go to 語(yǔ)句
goto語(yǔ)句稱為無(wú)條件轉(zhuǎn)移語(yǔ)句,通常與條件語(yǔ)句配合使用??捎脕?lái)實(shí)現(xiàn)條件轉(zhuǎn)移,?構(gòu)成循環(huán),跳出循環(huán)體等功能。但是,在結(jié)構(gòu)化程序設(shè)計(jì)中一般不主張使用goto語(yǔ)句,?以免造成程序流程的混亂,使理解和調(diào)試程序都產(chǎn)生困難。而且,goto語(yǔ)句一般可以用其他語(yǔ)句來(lái)代替。
當(dāng)然,goto語(yǔ)句也有其優(yōu)勢(shì),例如可以用goto語(yǔ)句一次性跳出多重循環(huán),而不需要使用多個(gè)break語(yǔ)句一步一步地跳出,從而使代碼更為簡(jiǎn)潔。
在C語(yǔ)言中使用goto語(yǔ)句的注意事項(xiàng)如下:
使用goto語(yǔ)句只能轉(zhuǎn)移到到同一函數(shù)內(nèi),而不能從一個(gè)函數(shù)里轉(zhuǎn)移到另外一個(gè)函數(shù)里。
使用goto語(yǔ)句在同一函數(shù)內(nèi)進(jìn)行轉(zhuǎn)移時(shí),轉(zhuǎn)移的起點(diǎn)應(yīng)是函數(shù)內(nèi)一段小功能的結(jié)束處,goto的目的label處應(yīng)是函數(shù)內(nèi)另外一段小功能的開(kāi)始處。
不能從一段復(fù)雜的執(zhí)行狀態(tài)中的位置goto到另外一個(gè)位置,比如,從多重嵌套的循環(huán)判斷中跳出去就是不允許的。
應(yīng)該避免向兩個(gè)方向跳轉(zhuǎn)。這樣最容易導(dǎo)致“面條代碼”。
所以說(shuō),goto語(yǔ)句要小心使用。
1樓答案完全錯(cuò)誤,有沒(méi)有用過(guò)goto?
建議樓主不要用goto語(yǔ)句,程序員最忌諱看goto程序,程序一復(fù)雜就很煩,這個(gè)程序還簡(jiǎn)單
先說(shuō)錯(cuò)誤原因:
else if (go==' '); //這里多了個(gè)分號(hào)";"所以一定執(zhí)行g(shù)oto end;
改進(jìn):去掉分號(hào)就可以了,像二樓那樣改達(dá)不到程序目的
程序中很多錯(cuò)誤,分析下:
#include stdio.h
#include conio.h
int zhong(int a, int b){return (a+b);}
int guo(int a, int b){return (a-b);}
int wo(int a, int b){return (a*b);}
int ni(int a, int b){return (a/b);}
int x,y,z,go; //go最好用char類型
main()
{
int zhong(int a, int b);
int guo(int a, int b);
int wo(int a, int b);
int ni(int a, int b);
get:
printf("please input one number!\n");
scanf("%c",go); //輸入時(shí)必定包含'字符'和'\r'(回車),這里只讀取字符,'\r'留在流中,下次運(yùn)行時(shí)會(huì)讀取'\r',建議:用getchar().
if(go=='+')
{
int zhong(int a, int b);
scanf("%d%d",x,y);
z=zhong(x,y);
if ((x||y)!=' ') //錯(cuò)誤,必定成立,因?yàn)?x||y)的值為1或0,永遠(yuǎn)不等于' '(空格字符)
{
printf ("%d+%d=%d\n",x,y,z);
}
else if ((x||y)==' ') //與上同理
{
printf("end\n");
}
}
else if (go=='-')
{
int guo(int a, int b);
scanf("%d%d",x,y);
z=guo(x,y);
if((x||y)!=' ') //error
{
printf ("%d-%d=%d\n",x,y,z);
}
else if ((x||y)==' ') //error
{
printf("end\n");
}
}
else if (go=='*')
{
int wo(int a, int b);
scanf("%d%d",x,y);
z=wo(x,y);
if((x||y)!=' ') //error
{
printf ("%d*%d=%d\n",x,y,z);
}
else if ((x||y)==' ') //error
{
printf("end\n");}
}
else if (go=='/')
{
int ni(int a, int b);
scanf("%d%d",x,y);
z=ni(x,y);
if ((x||y)!=' ') //error
{
printf ("%d/%d=%d\n",x,y,z);
}
else if ((x||y)==' ') //error
{
printf("end\n");
}
}
else if (go==' '); //錯(cuò)誤在這里,去掉";"就可以了
{
goto end;
}
goto get;
end:
printf("ByeBye!\n");
getch();
}
還有,除法會(huì)遇到圓整問(wèn)題,要x,y,z使用double類型
臭名昭著的goto出場(chǎng)了。
goto的漢義為“轉(zhuǎn)到”,在計(jì)算機(jī)語(yǔ)言里,它的完整名稱為:“無(wú)條件跳轉(zhuǎn)語(yǔ)句”。幾乎所有高級(jí)語(yǔ)言都會(huì)勸你盡量不要使用它goto。因?yàn)樗鼤?huì)破壞程序的模塊性,嚴(yán)重降低一段程序的可讀性。若是老外寫(xiě)的書(shū),則比喻使用大量goto的代碼:“像意大利面條”。嗯,其實(shí)北京的雜醬面也很纏繞……可惜沒(méi)有走向世界。
goto的用法是,首先要在代碼中某處加上一個(gè)位標(biāo)(也稱標(biāo)號(hào)),然后在代碼中的需處,加上goto,并寫(xiě)讓要跳轉(zhuǎn)到位標(biāo)。比如你在第三行代碼加一個(gè)位標(biāo):A : ,然后可以在第10行寫(xiě)上一個(gè)goto A,程序執(zhí)行到該行時(shí),就將跳到第三行。
加位標(biāo)的方法是在一空行加上位標(biāo)的名稱,命名規(guī)則和變量一樣,但最后要加上一冒號(hào)“:”。
例如:
int i = 1;
A :
cout i endl;
i++;
if(i = 10)
goto A;
... ...
goto 雖然號(hào)稱“無(wú)條件跳轉(zhuǎn)”,事實(shí)上倒是有些條件限制。主要是三條。
1、goto只能在當(dāng)前的同一程序段內(nèi)跳轉(zhuǎn);
2、goto 可以從循環(huán)內(nèi)跳轉(zhuǎn)到循環(huán)外的代碼,但不能從循環(huán)外的代碼跳到循環(huán)內(nèi);
3、在有g(shù)oto的跳轉(zhuǎn)范圍內(nèi),不能再使用C++允許的臨時(shí)變量聲明。
好了,其實(shí)筆者寫(xiě)程序近10年,惟一用到goto的地方就是:將一段簡(jiǎn)單的程序故意用goto寫(xiě)得面目全非,以期能讓破解程序的人因?yàn)檠蹠灦艞壒簟痪淅显挘喝绻麤](méi)有什么特殊理由,不要在程序里使用goto。
可能你會(huì)覺(jué)得沒(méi)有給出答案,但是看了以上的東西之后,我相信你會(huì)找到出錯(cuò)的地方,也許你已經(jīng)找著了。呵呵
goto語(yǔ)句也稱為無(wú)條件轉(zhuǎn)移語(yǔ)句,其一般格式如下: goto 語(yǔ)句標(biāo)號(hào); 其中語(yǔ)句標(biāo)號(hào)是按標(biāo)識(shí)符規(guī)定書(shū)寫(xiě)的符號(hào), 放在某一語(yǔ)句行的前面,標(biāo)號(hào)后加冒號(hào)(:)。語(yǔ)句標(biāo)號(hào)起標(biāo)識(shí)語(yǔ)句的作用,與goto 語(yǔ)句配合使用。
如: label: i++;
loop: while(x7);
goto loop;
C語(yǔ)言不限制程序中使用標(biāo)號(hào)的次數(shù),但各標(biāo)號(hào)不得重名。goto語(yǔ)句的語(yǔ)義是改變程序流向, 轉(zhuǎn)去執(zhí)行語(yǔ)句標(biāo)號(hào)所標(biāo)識(shí)的語(yǔ)句。
goto語(yǔ)句通常與條件語(yǔ)句配合使用??捎脕?lái)實(shí)現(xiàn)條件轉(zhuǎn)移, 構(gòu)成循環(huán),跳出循環(huán)體等功能。
擴(kuò)展資料:
go to語(yǔ)句使用原則:
1、使用goto語(yǔ)句只能goto到同一函數(shù)內(nèi),而不能從一個(gè)函數(shù)里goto到另外一個(gè)函數(shù)里。
2、使用goto語(yǔ)句在同一函數(shù)內(nèi)進(jìn)行g(shù)oto時(shí),goto的起點(diǎn)應(yīng)是函數(shù)內(nèi)一段小功能的結(jié)束處,goto的目的label處應(yīng)是函數(shù)內(nèi)另外一段小功能的開(kāi)始處。
3、不能從一段復(fù)雜的執(zhí)行狀態(tài)中的位置goto到另外一個(gè)位置,比如,從多重嵌套的循環(huán)判斷中跳出去就是不允許的。
4、應(yīng)該避免向兩個(gè)方向跳轉(zhuǎn)。這樣最容易導(dǎo)致"面條代碼"。
參考資料:百度百科-goto語(yǔ)句