golang請求參數驗證,golang函數默認參數

本文目錄一覽:

golang http請求數據怎麼傳body體參數

get請求一般是在你請求的地址後邊加上?par=”par2=”例如請求的頁面是a.do(假設你後台是java)那麼請求地址是這樣的a.do?par=’123’par2=’456’這個請求傳遞的參數就par和par2他倆的值分別為123,456

golang echo(二)—請求與響應

上一章節介紹了使用echo啟動一個http服務器並提供一個簡單的GET接口,這一章對請求與響應做一下詳細的講解。

了解請求與響應之前首先要知道ehco框架提供了一個Context接口,Context 表示當前 HTTP 請求的上下文,我們需要知道的是Context的實現結構體context,context中包含了請求(request)、響應(response)以及參數等結構體,context是一個內部結構體,但是對外提供了訪問的方法,我們可以在開發中使用context提供的方法做很多事情,比如獲取請求路徑、客戶端ip、請求參數等等。

request是包含了當前請求信息的結構體,我們從中可以獲取到前後端約定好的參數來作出相應的業務邏輯,這下面開始模擬請求

首先模擬一個添加用戶的POST請求,這裡使用了c.bind()這個方法,這個方法起到的作用是把表單參數或者JSON參數綁定至對應的結構體(這裡必須是指針類型),字段需要一一對應,也就是說前端需要傳遞的表單或JSON參數名為name、age、gender才能綁定到結構體中。

接下來模擬一個GET請求用來獲取用戶信息

response是包含了返回信息的結構體,echo提供了多種類型的返回結果

上面編寫了處理請求的handler,這裡介紹一下如何把handler註冊到路由中,這裡對應的就是上面handler。

golang net/http包 http請求的位元組碼讀取與解析。

先配置Header最長讀取時間、req最長讀取時間、req最大讀取長度默認6M。

RFC7230禁止\r\n參數,Url中只允許包含英文字母(a-zA-Z)、數字(0-9)、-_.~4個特殊字符以及所有保留字符。但go net/http包放寬了這個要求。

先構建newTextprotoReader,由於緩衝區是對象復用的,用完後要defer put。共完以以下解析任務:

TextprotoReader數據結構,將位元組碼Reader轉成文本Reader。

第一步,從第一行解析出method uri prototype。

第二步解析URL。url.URL數據結構:

解析Scheme,協議前綴(小寫)。有查詢參數?,則配置url.ForceQuery url.RawQuery。有認證信息///…//,則解析url.User url.Host。最後配置url.Path和url.RawPath,如果Path==RawPath,則RawPath=””。

第三步解析MIMEHeader。

第四步readTransfer。重新配置如下參數:RequestMethod ProtoMajor ProtoMinor Header Trailer ContentLength Close。對於Body,如果encodings支持chunked,讀取流用chunkedReader包裹。默認情況用LimitedReader,無body賦空的struct{}。

以下情況返回非空err,示得到正確的請求:

最後配置req.ctx req.RemoteAddr req.TLS body.doEarlyClose = true。

構建Response:

其中closeNotifyCh必須在構建時初始化,沒有content所以先置contentLength為-1。

配置w.cw並被w.w包裹。w.cw緩衝默認大小2M。

獲取Request可能出現如下錯誤:

先上響應數據結構:

response字段可以分類為:大對象、緩衝、KV對或bool型的狀態參數。

大對象有:

狀態字段:

chunkWriter數據結構:

chunkWriter包裹了Response,功能之一是完成Header設置,包括Content-Type Content-Length chunk-header。bufio.Writer是chunkWriter是緩衝包裹。

handler將響應寫入到response.w。

調用w.w.Flush()將w寫入到cw,注意到Flush()操作,如果未刷空緩存並報錯,觸發拷貝操作。報錯不會退回已寫出的數據。

進而調用cw.Write(),根據cw.chunking參數。

putBufioWriter(w.w)清空resp.w緩衝,如果池化放回sync.pool。

根據chunkWriter的定義,w.cw.close()負責cw的結束工作:寫入換行符和resp.trailers數據。

最後刷新TCP緩衝w.conn.bufw.Flush(),完成響應包發送。並正確關閉request。

golang jwt鑒權分析

技術棧 gin+jwt

鑒權流程:

調用token生成方法GenerateToken生成token, 請求api時帶上token參數即可

token計算邏輯:

總結:md5加密賬號和密碼參數,根據賬號密碼私鑰過期時間和jwt header 分別計算hash256值,將值用.符號連接,再進行hash,結果就是token值

api 驗證token:

Golang 中更好的錯誤處理:理論和實踐技巧

雲和安全管理服務專家新鈦雲服 張春翻譯

這種方法有幾個缺點。首先,它可以對程序員隱藏錯誤處理路徑,特別是在捕獲異常不是強制性的情況下,例如在 Python 中。即使在具有必須處理的 Java 風格的檢查異常的語言中,如果在與原始調用不同的級別上處理錯誤,也並不總是很明顯錯誤是從哪裡引發的。

我們都見過長長的代碼塊包裝在一個 try-catch 塊中。在這種情況下,catch 塊實際上充當 goto 語句,這通常被認為是有害的(奇怪的是,C 中的關鍵字被認為可以接受的少數用例之一是錯誤後清理,因為該語言沒有 Golang- 樣式延遲語句)。

如果你確實從源頭捕獲異常,你會得到一個不太優雅的 Go 錯誤模式版本。這可能會解決混淆代碼的問題,但會遇到另一個問題:性能。在諸如 Java 之類的語言中,拋出異常可能比函數的常規返回慢數百倍。

Java 中最大的性能成本是由打印異常的堆棧跟蹤造成的,這是昂貴的,因為運行的程序必須檢查編譯它的源代碼 。僅僅進入一個 try 塊也不是空閑的,因為需要保存 CPU 內存寄存器的先前狀態,因為它們可能需要在拋出異常的情況下恢復。

如果您將異常視為通常不會發生的異常情況,那麼異常的缺點並不重要。這可能是傳統的單體應用程序的情況,其中大部分代碼庫不必進行網絡調用——一個操作格式良好的數據的函數不太可能遇到錯誤(除了錯誤的情況)。一旦您在代碼中添加 I/O,無錯誤代碼的夢想就會破滅:您可以忽略錯誤,但不能假裝它們不存在!

try {

doSometing()

} catch (IOException e) {

// ignore it

}

與大多數其他編程語言不同,Golang 接受錯誤是不可避免的。 如果在單體架構時代還不是這樣,那麼在今天的模塊化後端服務中,服務通常和外部 API 調用、數據庫讀取和寫入以及與其他服務通信 。

以上所有方法都可能失敗,解析或驗證從它們接收到的數據(通常在無模式 JSON 中)也可能失敗。Golang 使可以從這些調用返回的錯誤顯式化,與普通返回值的等級相同。從函數調用返回多個值的能力支持這一點,這在大多數語言中通常是不可能的。Golang 的錯誤處理系統不僅僅是一種語言怪癖,它是一種將錯誤視為替代返回值的完全不同的方式!

重複 if err != nil

對 Go 錯誤處理的一個常見批評是被迫重複以下代碼塊:

res, err := doSomething()

if err != nil {

// Handle error

}

對於新用戶來說,這可能會覺得沒用而且浪費行數:在其他語言中需要 3 行的函數很可能會增長到 12 行 :

這麼多行代碼!這麼低效!如果您認為上述內容不優雅或浪費代碼,您可能忽略了我們檢查代碼中的錯誤的全部原因:我們需要能夠以不同的方式處理它們!對 API 或數據庫的調用可能會被重試。

有時事件的順序很重要:調用外部 API 之前發生的錯誤可能不是什麼大問題(因為數據從未通過發送),而 API 調用和寫入本地數據庫之間的錯誤可能需要立即注意,因為 這可能意味着系統最終處於不一致的狀態。即使我們只想將錯誤傳播給調用者,我們也可能希望用失敗的解釋來包裝它們,或者為每個錯誤返回一個自定義錯誤類型。

並非所有錯誤都是相同的,並且向調用者返回適當的錯誤是 API 設計的重要部分,無論是對於內部包還是 REST API 。

不必擔心在你的代碼中重複 if err != nil ——這就是 Go 中的代碼應該看起來的樣子。

自定義錯誤類型和錯誤包裝

從導出的方法返回錯誤時,請考慮指定自定義錯誤類型,而不是單獨使用錯誤字符串。字符串在意外代碼中是可以的,但在導出的函數中,它們成為函數公共 API 的一部分。更改錯誤字符串將是一項重大更改——如果沒有明確的錯誤類型,需要檢查返回錯誤類型的單元測試將不得不依賴原始字符串值!事實上,基於字符串的錯誤也使得在私有方法中測試不同的錯誤案例變得困難,因此您也應該考慮在包中使用它們。回到錯誤與異常的爭論,返回錯誤也使代碼比拋出異常更容易測試,因為錯誤只是要檢查的返回值。不需要測試框架或在測試中捕獲異常 。

可以在 database/sql 包中找到簡單自定義錯誤類型的一個很好的示例。它定義了一個導出常量列表,表示包可以返回的錯誤類型,最著名的是 sql.ErrNoRows。雖然從 API 設計的角度來看,這種特定的錯誤類型有點問題(您可能會爭辯說 API 應該返回一個空結構而不是錯誤),但任何需要檢查空行的應用程序都可以導入該常量並在代碼中使用它不必擔心錯誤消息本身會改變和破壞代碼。

對於更複雜的錯誤處理,您可以通過實現返回錯誤字符串的 Error() 方法來定義自定義錯誤類型。自定義錯誤可以包括元數據,例如錯誤代碼或原始請求參數。如果您想表示錯誤類別,它們很有用。DigitalOcean 的本教程展示了如何使用自定義錯誤類型來表示可以重試的一類臨時錯誤。

通常,錯誤會通過將低級錯誤與更高級別的解釋包裝起來,從而在程序的調用堆棧中傳播。例如,數據庫錯誤可能會以下列格式記錄在 API 調用處理程序中:調用 CreateUser 端點時出錯:查詢數據庫時出錯:pq:檢測到死鎖。這很有用,因為它可以幫助我們跟蹤錯誤在系統中傳播的過程,向我們展示根本原因(數據庫事務引擎中的死鎖)以及它對更廣泛系統的影響(調用者無法創建新用戶)。

自 Go 1.13 以來,此模式具有特殊的語言支持,並帶有錯誤包裝。通過在創建字符串錯誤時使用 %w 動詞,可以使用 Unwrap() 方法訪問底層錯誤。除了比較錯誤相等性的函數 errors.Is() 和 errors.As() 外,程序還可以獲取包裝錯誤的原始類型或標識。這在某些情況下可能很有用,儘管我認為在確定如何處理所述錯誤時最好使用頂級錯誤的類型。

Panics

不要 panic()!長時間運行的應用程序應該優雅地處理錯誤而不是panic。即使在無法恢復的情況下(例如在啟動時驗證配置),最好記錄一個錯誤並優雅地退出。panic比錯誤消息更難診斷,並且可能會跳過被推遲的重要關閉代碼。

Logging

我還想簡要介紹一下日誌記錄,因為它是處理錯誤的關鍵部分。通常你能做的最好的事情就是記錄收到的錯誤並繼續下一個請求。

除非您正在構建簡單的命令行工具或個人項目,否則您的應用程序應該使用結構化的日誌庫,該庫可以為日誌添加時間戳,並提供對日誌級別的控制。最後一部分特別重要,因為它將允許您突出顯示應用程序記錄的所有錯誤和警告。通過幫助將它們與信息級日誌分開,這將為您節省無數時間。

微服務架構還應該在日誌行中包含服務的名稱以及機器實例的名稱。默認情況下記錄這些時,程序代碼不必擔心包含它們。您也可以在日誌的結構化部分中記錄其他字段,例如收到的錯誤(如果您不想將其嵌入日誌消息本身)或有問題的請求或響應。只需確保您的日誌沒有泄露任何敏感數據,例如密碼、API 密鑰或用戶的個人數據!

對於日誌庫,我過去使用過 logrus 和 zerolog,但您也可以選擇其他結構化日誌庫。如果您想了解更多信息,互聯網上有許多關於如何使用這些的指南。如果您將應用程序部署到雲中,您可能需要日誌庫上的適配器來根據您的雲平台的日誌 API 格式化日誌 – 沒有它,雲平台可能無法檢測到日誌級別等某些功能。

如果您在應用程序中使用調試級別日誌(默認情況下通常不記錄),請確保您的應用程序可以輕鬆更改日誌級別,而無需更改代碼。更改日誌級別還可以暫時使信息級別甚至警告級別的日誌靜音,以防它們突然變得過於嘈雜並開始淹沒錯誤。您可以使用在啟動時檢查以設置日誌級別的環境變量來實現這一點。

原文:

原創文章,作者:小藍,如若轉載,請註明出處:https://www.506064.com/zh-hk/n/201115.html

(0)
打賞 微信掃一掃 微信掃一掃 支付寶掃一掃 支付寶掃一掃
小藍的頭像小藍
上一篇 2024-12-06 11:30
下一篇 2024-12-06 11:30

相關推薦

  • Python中引入上一級目錄中函數

    Python中經常需要調用其他文件夾中的模塊或函數,其中一個常見的操作是引入上一級目錄中的函數。在此,我們將從多個角度詳細解釋如何在Python中引入上一級目錄的函數。 一、加入環…

    編程 2025-04-29
  • Python中capitalize函數的使用

    在Python的字符串操作中,capitalize函數常常被用到,這個函數可以使字符串中的第一個單詞首字母大寫,其餘字母小寫。在本文中,我們將從以下幾個方面對capitalize函…

    編程 2025-04-29
  • Python中set函數的作用

    Python中set函數是一個有用的數據類型,可以被用於許多編程場景中。在這篇文章中,我們將學習Python中set函數的多個方面,從而深入了解這個函數在Python中的用途。 一…

    編程 2025-04-29
  • 單片機打印函數

    單片機打印是指通過串口或並口將一些數據打印到終端設備上。在單片機應用中,打印非常重要。正確的打印數據可以讓我們知道單片機運行的狀態,方便我們進行調試;錯誤的打印數據可以幫助我們快速…

    編程 2025-04-29
  • 三角函數用英語怎麼說

    三角函數,即三角比函數,是指在一個銳角三角形中某一角的對邊、鄰邊之比。在數學中,三角函數包括正弦、餘弦、正切等,它們在數學、物理、工程和計算機等領域都得到了廣泛的應用。 一、正弦函…

    編程 2025-04-29
  • 三星內存條參數用法介紹

    本文將詳細解釋三星內存條上面的各種參數,讓你更好地了解內存條並選擇適合自己的一款。 一、容量大小 容量大小是內存條最基本的參數,一般以GB為單位表示,常見的有2GB、4GB、8GB…

    編程 2025-04-29
  • Python3定義函數參數類型

    Python是一門動態類型語言,不需要在定義變量時顯示的指定變量類型,但是Python3中提供了函數參數類型的聲明功能,在函數定義時明確定義參數類型。在函數的形參後面加上冒號(:)…

    編程 2025-04-29
  • Python實現計算階乘的函數

    本文將介紹如何使用Python定義函數fact(n),計算n的階乘。 一、什麼是階乘 階乘指從1乘到指定數之間所有整數的乘積。如:5! = 5 * 4 * 3 * 2 * 1 = …

    編程 2025-04-29
  • Python定義函數判斷奇偶數

    本文將從多個方面詳細闡述Python定義函數判斷奇偶數的方法,並提供完整的代碼示例。 一、初步了解Python函數 在介紹Python如何定義函數判斷奇偶數之前,我們先來了解一下P…

    編程 2025-04-29
  • Spring Boot中發GET請求參數的處理

    本文將詳細介紹如何在Spring Boot中處理GET請求參數,並給出完整的代碼示例。 一、Spring Boot的GET請求參數基礎 在Spring Boot中,處理GET請求參…

    編程 2025-04-29

發表回復

登錄後才能評論