gin框架學習筆記

Gin 是一個基于 Go 語言的高性能 Web 框架

gin下載

在已有的go項目直接終端輸入

go get -u github.com/gin-gonic/gin

hello world快速上手

package mainimport ("github.com/gin-gonic/gin"
)func main() {router := gin.Default()router.GET("/", func(c *gin.Context) {c.JSON(200, gin.H{"message": "Hello world!",})})router.Run()
}

運行

go run main.go

瀏覽器輸入http://localhost:8080/

可以看到瀏覽器顯示

到這里簡單的gin上手完成了

參數獲取

路徑參數獲取

這里可以類比Java springboot 的@RequestParam

即獲取路徑url?后邊的參數數據

func main() {r := gin.Default()r.GET("/user/phone", func(c *gin.Context) {//設置默認的數值 假設路徑參數phone沒有傳值則給一個默認值phone := c.DefaultQuery("phone", "333")//直接獲取路徑參數username := c.Query("username")Json格式返回參數c.JSON(http.StatusOK, gin.H{"message":  "ok","username": username,"phone":    phone,})})r.Run()
}

瀏覽器輸入user/phone?phone=333&username=用戶?

瀏覽器輸入user/phone?username=用戶

路徑參數沒有給phone的參數

這里給出了默認的返回333

這里的返回是由于gin程序里 phone := c.DefaultQuery("phone", "333")實現的

form參數獲取

func main() {//Default返回一個默認的路由引擎r := gin.Default()r.POST("/user/phone", func(c *gin.Context) {// DefaultPostForm取不到值時會返回指定的默認值username := c.DefaultPostForm("username", "用戶")// username := c.PostForm("username")phone := c.PostForm("phone")c.JSON(http.StatusOK, gin.H{"message":  "ok","username": username,"phone":    phone,})})r.Run(":8080")
}

postman調試(參數都有傳值)

返回

postman調試(username參數沒有傳值)

返回默認值 “用戶”

這里實現和獲取默認路徑差不多??? ??? ?username := c.DefaultPostForm("username", "用戶")

調用DefaultPostForm實現的

json參數獲取

這里可以類比Java的@RequestBody

如下程序實現了把給定的json參數原樣返回

func main() {r := gin.Default()r.POST("/json", func(c *gin.Context) {b, err := c.GetRawData() // 從c.Request.Body讀取請求數據if err != nil {log.Fatal(err)}var m map[string]interface{}// 反序列化err1 := json.Unmarshal(b, &m)if err1 != nil {log.Fatal(err1)}c.JSON(http.StatusOK, m)})r.Run(":8080")
}

postman調用

返回

path參數獲取

可以類比成Java的@PathVariable

func main() {r := gin.Default()r.GET("/user/phone/:username/:phone", func(c *gin.Context) {username := c.Param("username")phone := c.Param("phone")//輸出json結果給調用方c.JSON(http.StatusOK, gin.H{"message":  "ok","username": username,"phone":    phone,})})r.Run(":8080")
}

postman調用

傳格式如下

/user/phone/測試用戶參數/10190010933

返回

參數綁定

即利用反射機制,通過請求的Content-Type以及ShouldBind函數自動判斷請求的格式來解析出來再綁定到返回的數據中,這里就和springboot的@RequestParam等注解自動解析差不多

type Login struct {User     string `form:"user" json:"user" binding:"required"`Password string `form:"password" json:"password" binding:"required"`
}func main() {router := gin.Default()// 綁定JSON的示例 ({"user": "q1mi", "password": "123456"})router.POST("/loginJSON", func(c *gin.Context) {var login Loginif err := c.ShouldBind(&login); err == nil {fmt.Printf("login info:%#v\n", login)c.JSON(http.StatusOK, gin.H{"user":     login.User,"password": login.Password,})} else {c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})}})// 綁定form表單示例 (user=q1mi&password=123456)router.POST("/loginForm", func(c *gin.Context) {var login Login// ShouldBind()會根據請求的Content-Type自行選擇綁定器if err := c.ShouldBind(&login); err == nil {c.JSON(http.StatusOK, gin.H{"user":     login.User,"password": login.Password,})} else {c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})}})// 綁定QueryString示例 (/loginQuery?user=q1mi&password=123456)router.GET("/loginQuery", func(c *gin.Context) {var login Login// ShouldBind()會根據請求的Content-Type自行選擇綁定器if err := c.ShouldBind(&login); err == nil {c.JSON(http.StatusOK, gin.H{"user":     login.User,"password": login.Password,})} else {c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})}})// Listen and serve on 0.0.0.0:8080router.Run(":8080")
}

json格式

返回

form格式

返回

query格式

返回

文件上傳

單文件上傳

func main() {router := gin.Default()// 處理multipart forms提交文件時默認的內存限制是32 MiB// 可以通過下面的方式修改// router.MaxMultipartMemory = 8 << 20  // 8 MiBrouter.POST("/upload", func(c *gin.Context) {// 單個文件file, err := c.FormFile("f1")if err != nil {c.JSON(http.StatusInternalServerError, gin.H{"message": err.Error(),})return}log.Println(file.Filename)dst := fmt.Sprintf("/Users/kanyu/Desktop/project/kanyu_server/ginlearn/templates/user/%s", file.Filename)// 上傳文件到指定的目錄c.SaveUploadedFile(file, dst)c.JSON(http.StatusOK, gin.H{"message": fmt.Sprintf("'%s' uploaded!", file.Filename),})})router.Run()
}

原項目結構

postman上傳文件

結果

可以看到文件上傳成功 目錄下多了個3.zip文件

多文件上傳

func main() {router := gin.Default()// 處理multipart forms提交文件時默認的內存限制是32 MiB// 可以通過下面的方式修改// router.MaxMultipartMemory = 8 << 20  // 8 MiBrouter.POST("/upload", func(c *gin.Context) {// Multipart formform, _ := c.MultipartForm()//取key=file的參數文件files := form.File["file"]for index, file := range files {log.Println(file.Filename)dst := fmt.Sprintf("/Users/kanyu/Desktop/project/kanyu_server/ginlearn/templates/user/_%d_%s", index, file.Filename)// 上傳文件到指定的目錄c.SaveUploadedFile(file, dst)}c.JSON(http.StatusOK, gin.H{"message": fmt.Sprintf("%d files uploaded!", len(files)),})})router.Run()
}

原項目結構

postman調用多文件上傳

返回

項目結構

路由

常規路由

router.GET("/", func(c *gin.Context) {}router.POST("/", func(c *gin.Context) {}router.GET("/", func(c *gin.Context) {}

這些路由在之前參數獲取的時候講過了

和java中差不多 根據請求類型和路徑的不同獲取請求的參數進行處理

Any

func main() {router := gin.Default()router.Any("/", func(c *gin.Context) {c.JSON(200, gin.H{"message": "Hello world!",})})router.Run()
}

get請求

post請求

可以看到無論請求類型如何 都會返回程序中定義的"Hello world!"

NoRoute

即無路由請求傳任何路徑的返回處理

func main() {router := gin.Default()router.NoRoute(func(c *gin.Context) {c.JSON(200, gin.H{"message": "404 Not Found",})})router.Run()
}

給定任意路徑

路由組

相當于Java springboot的如下結構 利用@RestController和@RequestMapping("/chat")

都擁有共同的前綴/chat

所以擁有共同的前綴url劃分為路由組

func main() {r := gin.Default()userGroup := r.Group("/user"){userGroup.GET("/index", func(c *gin.Context) {c.JSON(200, gin.H{"message": "user index",})})userGroup.GET("/login", func(c *gin.Context) {c.JSON(200, gin.H{"message": "user login GET",})})userGroup.POST("/login", func(c *gin.Context) {c.JSON(200, gin.H{"message": "user login POST",})})}shopGroup := r.Group("/shop"){shopGroup.GET("/index", func(c *gin.Context) {c.JSON(200, gin.H{"message": "shop login",})})shopGroup.GET("/cart", func(c *gin.Context) {c.JSON(200, gin.H{"message": "shop cart",})})shopGroup.POST("/checkout", func(c *gin.Context) {c.JSON(200, gin.H{"message": "shop checkout",})})// 嵌套路由組shouIndexGroup := shopGroup.Group("/index")shouIndexGroup.GET("/router", func(c *gin.Context) {c.JSON(200, gin.H{"message": "shop index router",})})}r.Run()
}

請求/user/index

請求GET? /user/login

請求POST??/user/login

請求/shop/index

請求/shop/cart

請求/shop/checkout

嵌套路由?請求/shop/index/router

中間件

專注于HTTP請求處理流程的攔截與擴展,如日志記錄、權限驗證、異常恢復等。其核心是通過鏈式調用模式對單個請求生命周期進行功能增強,屬于輕量級、高并發的Web開發組件,
比如Gin默認中間件gin.Logger()用于請求日志記錄,gin.Recovery()用于panic恢復

gin框架的中間件類比Java

如Spring Interceptor的preHandle()postHandle()方法,與Gin中間件的請求前后邏輯對應,適合權限檢查、參數預處理等場景

如Java?Servlet Filter均通過鏈式調用(Chain of Responsibility模式)攔截HTTP請求,可在請求處理前、后插入邏輯

定義中間件和注冊

func StatCost() gin.HandlerFunc {return func(c *gin.Context) {start := time.Now()c.Set("name", "小王子") // 可以通過c.Set在請求上下文中設置值,后續的處理函數能夠取到該值// 調用該請求的剩余處理程序c.Next()// 不調用該請求的剩余處理程序// c.Abort()// 計算耗時cost := time.Since(start)log.Println(cost)}
}// 全局注冊路由
func main() {// 新建一個沒有任何默認中間件的路由r := gin.New()// 注冊一個全局中間件r.Use(StatCost())r.GET("/test", func(c *gin.Context) {name := c.MustGet("name").(string) // 從上下文取值log.Println(name)c.JSON(http.StatusOK, gin.H{"message": "Hello " + name,})})r.Run()
}

調用/test

可以看到返回了中間件前置處理塞進去的值 "小王子"

這里利用c.MustGet("name").(string)取到了上下文數值,底層利用了Context的能力,這個后續再記錄下Context的核心實現和原理

單個路由注冊中間件

func StatCost() gin.HandlerFunc {return func(c *gin.Context) {start := time.Now()c.Set("name", "小王子") // 可以通過c.Set在請求上下文中設置值,后續的處理函數能夠取到該值// 調用該請求的剩余處理程序c.Next()// 不調用該請求的剩余處理程序// c.Abort()// 計算耗時cost := time.Since(start)log.Println(cost)}
}//記錄響應體中間件type bodyLogWriter struct {gin.ResponseWriter               // 嵌入gin框架ResponseWriterbody               *bytes.Buffer // 我們記錄用的response
}// Write 寫入響應體數據
func (w bodyLogWriter) Write(b []byte) (int, error) {w.body.Write(b)                  // 我們記錄一份return w.ResponseWriter.Write(b) // 真正寫入響應
}// ginBodyLogMiddleware 一個記錄返回給客戶端響應體的中間件
// https://stackoverflow.com/questions/38501325/how-to-log-response-body-in-gin
func ginBodyLogMiddleware() gin.HandlerFunc {return func(c *gin.Context) {blw := &bodyLogWriter{body: bytes.NewBuffer([]byte{}), ResponseWriter: c.Writer}c.Writer = blw // 使用我們自定義的類型替換默認的c.Next() // 執行業務邏輯fmt.Println("Response body: " + blw.body.String()) // 事后按需記錄返回的響應}
}func main() {// 新建一個沒有任何默認中間件的路由r := gin.New()// 注冊兩個中間件 一個記錄耗時 一個記錄響應體 這里可以注冊多個中間件r.GET("/test", StatCost(), ginBodyLogMiddleware(), func(c *gin.Context) {name := c.MustGet("name").(string) // 從上下文取值log.Println(name)c.JSON(http.StatusOK, gin.H{"message": "Hello " + name,})})r.Run()
}

請求

后臺打印日志

可以看到后臺打印出了相應體?Response body: {"message":"Hello 小王子"}

以及耗時209.369

參考:

李文周go語言教程

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/77756.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/77756.shtml
英文地址,請注明出處:http://en.pswp.cn/web/77756.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

linux中由于編譯選項-D_OS64BIT導致的核心已轉儲問題

linux中由于編譯選項-D_OS64BIT導致的核心已轉儲問題排查解決&#xff1a; 原因&#xff1a; a.so b.so a.so使用b.so 程序1 程序2 使用a.so 程序1運行正常&#xff0c;程序2啟動后提示核心已轉儲。 程序1和程序2運行的代碼都一致&#xff0c;只執行創建xApplication app&…

什么是ICSP編程

ICSP編程介紹 ICSP 編程&#xff08;In-Circuit Serial Programming&#xff09;&#xff0c;即“在線串行編程”&#xff0c;是一種通過 SPI 協議 直接對微控制器&#xff08;如 Arduino 的 ATmega328P&#xff09;進行編程的技術&#xff0c;無需移除芯片。它常用于以下場景…

基于Vue3和OpenLayers的WebGIS示例程序

筆記參考教程來源于B站UP主znlgis的視頻合集&#xff1a;https://space.bilibili.com/161342702&#xff0c;直播使用的源碼地址&#xff1a;https://github.com/OpenGisToolbox。 Demo合集分為5大部分&#xff0c;分別是&#xff1a;基礎環境搭建、項目搭建、GeoServer Rest A…

UBUS 通信接口的使用——添加一個object對象(ubus call)

1&#xff0c;引入 ubus提供了一種多進程通信的機制。存在一個守護進程ubusd&#xff0c;所以進程都注冊到ubusd&#xff0c;ubusd進行消息的接收、分發管理。 ubus對多線程支持的不好&#xff0c;例如在多個線程中去請求同一個服務&#xff0c;就有可能出現不可預知的結果。 …

【Python魔法方法(特殊方法)】

在 Python 中&#xff0c;許多運算符都可以進行重載&#xff0c;以下是一些常見運算符及其對應的魔法方法&#xff08;特殊方法&#xff09;&#xff1a; 算術運算符 加法 &#xff1a;__add__ 用于定義對象相加的行為。例如&#xff0c;當你對兩個自定義類的實例使用 運算符…

(三十二)Android開發中AppCompatActivity和Activity之間的詳細區別

在 Android 開發中&#xff0c;AppCompatActivity 和 Activity 是兩個核心類&#xff0c;用于創建和管理應用程序的用戶界面。盡管它們功能上有重疊&#xff0c;但它們之間存在顯著的區別。本文將詳細講解 AppCompatActivity 和 Activity 的區別&#xff0c;并結合代碼示例和具…

【 C++核心知識點面試準備:從內存管理到STL與模板 】

一、動態內存管理&#xff1a;new/delete與底層原理 核心問題1&#xff1a;new/delete vs malloc/free 區別對比&#xff1a; 特性new/deletemalloc/free類型安全自動推導類型&#xff0c;無需轉型返回void*&#xff0c;需強制轉型生命周期自動調用構造/析構函數需手動初始化…

軟考高項(信息系統項目管理師)第 4 版全章節核心考點解析(第4版課程精華版)

一、核心輸入輸出速記體系&#xff08;力揚老師獨家口訣&#xff09; &#xff08;一&#xff09;規劃階段萬能輸入&#xff08;4 要素&#xff09; 口訣&#xff1a;章程計劃&#xff0c;組織事業 ? 精準對應&#xff08;ITTO 核心輸入&#xff09;&#xff1a; 章程&#…

ASP.NET CORE部署IIS的三種方式

ASP.NET Core 部署方式對比 本文檔對比了三種常見的 ASP.NET Core 應用&#xff08;如你的 DingTalkApproval 項目&#xff09;部署到 Windows 10 上 IIS 服務器的方式&#xff1a;dotnet publish&#xff08;手動部署&#xff09;、Web Deploy&#xff08;直接發布到 IIS&…

基于共享上下文和自主協作的 RD Agent 生態系統

在llmangentmcp這個框架中&#xff1a; LLM&#xff1a; 依然是智能體的“大腦”&#xff0c;賦予它們理解、推理、生成和規劃的能力&#xff0c;并且也用于處理和利用共享上下文。Agent&#xff1a; 具備特定 R&D 職能的自主單元&#xff0c;它們感知共享上下文&#xff0…

zephyr架構下Bluetooth advertising接口

目錄 概述 1 函數接口 2 主要函數介紹 2.1 bt_le_adv_start函數 2.1.1 函數功能介紹 2.1.2 典型使用示例 2.1.3 廣播間隔 2.1.4 注意事項 2.2 bt_le_adv_stop 函數 2.2.1 函數功能 2.2.2 使用方法介紹 2.2.3 實際應用示例 2.2.4 關鍵注意事項 2.2.5 常見問題解決 …

8、HTTPD服務--ab壓力測試

一、ab壓力測試 # ab ‐c 100 ‐n 1000 http://vedio.linux.com/index.html 2 This is ApacheBench, Version 2.3 <$Revision: 1430300 $> 3 Copyright 1996 Adam Twiss, Zeus Technology Ltd, http://www.zeustech.net/ 4 Licensed to The Apache Software Foundation,…

E2E 測試

以下是關于端到端(E2E)測試的基本知識總結: 一、E2E 測試核心認知 1. 定義與價值定位 "模擬真實用戶在完整應用環境中的操作流程"核心價值: 驗證跨系統/模塊的集成功能檢測用戶流程中的關鍵路徑保障核心業務場景的可用性測試金字塔定位:單元測試(70%) → 集…

python之數字類型的操作

Python數據類型與操作符完全指南&#xff1a;詳解各類數據操作技巧 目錄 數字類型 字符串 列表 元組 字典 集合 布爾 通用操作符 注意事項 1. 數字類型&#xff08;int, float, complex&#xff09; 數字類型是Python中最基礎的數據類型&#xff0c;支持多種數學運算…

基于Spring Boot+Vue 網上書城管理系統設計與實現(源碼+文檔+部署講解)

技術范圍&#xff1a;SpringBoot、Vue、SSM、HLMT、Jsp、PHP、Nodejs、Python、爬蟲、數據可視化、小程序、安卓app、大數據、物聯網、機器學習等設計與開發。 主要內容&#xff1a;免費功能設計、開題報告、任務書、中期檢查PPT、系統功能實現、代碼編寫、論文編寫和輔導、論文…

從拒絕采樣到強化學習,大語言模型推理極簡新路徑!

大語言模型&#xff08;LLMs&#xff09;的推理能力是當下研究熱點&#xff0c;強化學習在其復雜推理任務微調中廣泛應用。這篇論文深入剖析了相關算法&#xff0c;發現簡單的拒絕采樣基線方法表現驚人&#xff0c;還提出了新算法。快來一探究竟&#xff0c;看看這些發現如何顛…

測試——BUG篇

1. 軟件測試的生命周期 軟件測試貫穿于軟件的整個生命周期&#xff0c;針對這句話我們?起來看?下軟件測試是如何貫穿軟件的整個生命周期。 軟件測試的?命周期是指測試流程&#xff0c;這個流程是按照?定順序執?的?系列特定的步驟&#xff0c;去保證產品質量符合需求。在…

【Hive入門】Hive函數:內置函數與UDF開發

Apache Hive作為Hadoop生態系統中的重要組件&#xff0c;為大數據分析提供了強大的SQL-like查詢能力。Hive不僅支持豐富的內置函數&#xff0c;還允許用戶開發自定義函數&#xff08;UDF&#xff09;以滿足特定需求。本文將深入探討Hive的內置函數&#xff08;包括數學函數、字…

關于匯編語言與程序設計——子程序設計

學習目標&#xff1a; 編程實現兩個數&#xff1a;#8888H 和 #79H 的乘除運算。 一、實驗要求 能夠熟練掌握算術運算匯編指令的使用&#xff1b;熟練掌握子程序設計的基本方法&#xff1b;熟練掌握程序的調試方法。 二、實驗設計 1.整體思路 乘法&#xff1a;將單字節的乘數…

AWS SQS 隊列策略配置指南:常見錯誤與解決方案

在 AWS 云服務中,Simple Queue Service (SQS) 是一種完全托管的消息隊列服務,廣泛應用于分布式系統組件間的解耦。為了確保隊列的安全訪問,正確配置隊列策略至關重要。本文將詳細介紹 SQS 隊列策略的配置方法,常見錯誤及其解決方案。 SQS 隊列策略基礎 SQS 隊列策略是基于…