go get -u github.com/gin-gonic/gin
一個簡單的例子:package main import "github.com/gin-gonic/gin" func main() { //Default返回一個預設的路由引擎 r := gin.Default() r.GET("/ping", func(c *gin.Context) { //輸出json結果給呼叫方 c.JSON(200, gin.H{ "message": "pong", }) }) r.Run() // listen and serve on 0.0.0.0:8080 }編譯執行程式,開啟瀏覽器,存取
http://localhost:8080/ping
頁面顯示:{"message":"pong"}
gin 的功能不只是簡單輸出 Json 資料。它是一個輕量級的 WEB 框架,支援 RestFull 風格 API,支援 GET,POST,PUT,PATCH,DELETE,OPTIONS 等 http 方法,支援檔案上傳,分組路由,Multipart/Urlencoded FORM,以及支援 JsonP,引數處理等等功能,這些都和 WEB 緊密相關,通過提供這些功能,使開發人員更方便地處理 WEB 業務。package main import ( "net/http" "github.com/gin-gonic/gin" ) func main() { router := gin.Default() // 靜態資源載入,本例為css,js以及資源圖片 router.StaticFS("/public", http.Dir("D:/goproject/src/github.com/ffhelicopter/tmm/website/static")) router.StaticFile("/favicon.ico", "./resources/favicon.ico") // Listen and serve on 0.0.0.0:80 router.Run(":80") }首先需要是生成一個 Engine,這是 gin 的核心,預設帶有 Logger 和 Recovery 兩個中介軟體。
router := gin.Default()
StaticFile 是載入單個檔案,而 StaticFS 是載入一個完整的目錄資源:
func (group *RouterGroup) StaticFile(relativePath, filepath string) IRoutes
func (group *RouterGroup) StaticFS(relativePath string, fs http.FileSystem) IRoutes
http://localhost/public/images/logo.jpg
圖片載入正常。每次請求響應都會在伺服器端有紀錄檔產生,包括響應時間,載入資源名稱,響應狀態值等等。package main import ( "context" "log" "net/http" "os" "os/signal" "time" "github.com/ffhelicopter/tmm/handler" "github.com/gin-gonic/gin" ) func main() { router := gin.Default() // 靜態資源載入,本例為css,js以及資源圖片 router.StaticFS("/public", http.Dir("D:/goproject/src/github.com/ffhelicopter/tmm/website/static")) router.StaticFile("/favicon.ico", "./resources/favicon.ico") // 匯入所有模板,多級目錄結構需要這樣寫 router.LoadHTMLGlob("website/tpl/*/*") // website分組 v := router.Group("/") { v.GET("/index.html", handler.IndexHandler) v.GET("/add.html", handler.AddHandler) v.POST("/postme.html", handler.PostmeHandler) } // router.Run(":80") // 這樣寫就可以了,下面所有程式碼(go1.8+)是為了優雅處理重新啟動等動作。 srv := &http.Server{ Addr: ":80", Handler: router, ReadTimeout: 30 * time.Second, WriteTimeout: 30 * time.Second, } go func() { // 監聽請求 if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed { log.Fatalf("listen: %sn", err) } }() // 優雅Shutdown(或重新啟動)服務 quit := make(chan os.Signal) signal.Notify(quit, os.Interrupt) // syscall.SIGKILL <-quit log.Println("Shutdown Server ...") ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) defer cancel() if err := srv.Shutdown(ctx); err != nil { log.Fatal("Server Shutdown:", err) } select { case <-ctx.Done(): } log.Println("Server exiting") }在動態站點實現中,引入 WEB 分組以及優雅重新啟動這兩個功能。WEB 分組功能可以通過不同的入口根路徑來區別不同的模組,這裡我們可以存取:
http://localhost/index.html
。如果新增一個分組,比如:v := router.Group("/login")
我們可以存取:http://localhost/login/xxxx
,xxx 是我們在 v.GET 方法或 v.POST 方法中的路徑。
// 匯入所有模板,多級目錄結構需要這樣寫
router.LoadHTMLGlob("website/tpl/*/*")
// website分組
v := router.Group("/")
{
v.GET("/index.html", handler.IndexHandler)
v.GET("/add.html", handler.AddHandler)
v.POST("/postme.html", handler.PostmeHandler)
}
router.LoadHTMLGlob("website/tpl/*/*")
比如 v.GET("/index.html", handler.IndexHandler),通過存取http://localhost/index.html
這個 URL,實際由 handler.IndexHandler 來處理。而在 tmm 目錄下的 handler 存放了 package handler 檔案。在包裡定義了 IndexHandler 函數,它使用了 index.html 模板。
func IndexHandler(c *gin.Context) {
c.HTML(http.StatusOK, "index.html", gin.H{
"Title": "作品欣賞",
})
}
<!DOCTYPE html> <html> <head> {{template "header" .}} </head> <body> <!--導航--> <div class="feeds"> <div class="top-nav"> <a href="/index.tml" class="active">欣賞</a> <a href="/add.html" class="add-btn"> <svg class="icon" aria-hidden="true"> <use xlink:href="#icon-add"></use> </svg> 發布 </a> </div> <input type="hidden" id="showmore" value="{$showmore}"> <input type="hidden" id="page" value="{$page}"> <!--</div>--> </div> <script type="text/javascript"> var done = true; $(window).scroll(function(){ var scrollTop = $(window).scrollTop(); var scrollHeight = $(document).height(); var windowHeight = $(window).height(); var showmore = $("#showmore").val(); if(scrollTop + windowHeight + 300 >= scrollHeight && showmore == 1 && done){ var page = $("#page").val(); done = false; $.get("{:U('Product/listsAjax')}", { page : page }, function(json) { if (json.rs != "") { $(".feeds").append(json.rs); $("#showmore").val(json.showmore); $("#page").val(json.page); done = true; } },'json'); } }); </script> <script src="//at.alicdn.com/t/font_ttszo9rnm0wwmi.js"></script> </body> </html>在 index.html 模板中,通過 {{template "header" .}} 語句,巢狀了 header.html 模板。
{{ define "header" }} <meta charset="UTF-8"> <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1, minimum-scale=1, user-scalable=no, minimal-ui"> <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> <meta name="format-detection" content="telephone=no,email=no"> <title>{{ .Title }}</title> <link rel="stylesheet" href="/public/css/common.css"> <script src="/public/lib/jquery-3.1.1.min.js"></script> <script src="/public/lib/jquery.cookie.js"></script> <link href="/public/css/font-awesome.css?v=4.4.0" rel="stylesheet"> {{ end }}{{ define "header" }} 讓我們在模板巢狀時直接使用 header 名字,而在 index.html 中的 {{template "header" .}} 注意“.”,可以使引數巢狀傳遞,否則不能傳遞,比如這裡的 Title。
http://localhost/index.html
,可以看到瀏覽器顯示 Title 是“作品欣賞”,這個 Title 是通過 IndexHandler 來指定的。有關 IPFS: IPFS 本質上是一種內容可定址、版本化、對等超媒體的分散式儲存、傳輸協定,目標是補充甚至取代過去 20 年裡使用的超文字媒體傳輸協定(HTTP),希望構建更快、更安全、更自由的網際網路時代。注意:由於在本人在發佈到 github 的程式碼中,在處理圖片上傳的程式碼中,除了伺服器儲存外,還實現了 IPFS 發佈儲存,如果不需要 IPFS,請註釋相關程式碼。
import "github.com/didip/tollbooth/limiter"
包,在上面程式碼基礎上增加如下語句:
//rate-limit 限流中介軟體
lmt := tollbooth.NewLimiter(1, nil)
lmt.SetMessage("服務繁忙,請稍後再試...")
v.GET("/index.html", LimitHandler(lmt), handler.IndexHandler)
當 F5 重新整理重新整理http://localhost/index.html
頁面時,瀏覽器會顯示:服務繁忙,請稍後再試...tollbooth.LimitByKeys(lmt, []string{"127.0.0.1", "/"})
更多限流策略的設定,可以進一步github.com/didip/tollbooth/limiter
了解。http://localhost/v1/user/1100000/
這裡對 v1.GET("/user/:id/*action", LimitHandler(lmt), api.GetUser) 進行了限流控制,所以如果頻繁存取上面地址也將會有限制,這在 API 介面中非常有作用。
//CORS 區域性CORS,可在路由中設定全域性的CORS
c.Writer.Header().Add("Access-Control-Allow-Origin", "*")
package main import ( "context" "log" "net/http" "os" "os/signal" "time" "github.com/didip/tollbooth" "github.com/didip/tollbooth/limiter" "github.com/ffhelicopter/tmm/api" "github.com/ffhelicopter/tmm/handler" "github.com/gin-gonic/gin" ) // 定義全域性的CORS中介軟體 func Cors() gin.HandlerFunc { return func(c *gin.Context) { c.Writer.Header().Add("Access-Control-Allow-Origin", "*") c.Next() } } func LimitHandler(lmt *limiter.Limiter) gin.HandlerFunc { return func(c *gin.Context) { httpError := tollbooth.LimitByRequest(lmt, c.Writer, c.Request) if httpError != nil { c.Data(httpError.StatusCode, lmt.GetMessageContentType(), []byte(httpError.Message)) c.Abort() } else { c.Next() } } } func main() { gin.SetMode(gin.ReleaseMode) router := gin.Default() // 靜態資源載入,本例為css,js以及資源圖片 router.StaticFS("/public", http.Dir("D:/goproject/src/github.com/ffhelicopter/tmm/website/static")) router.StaticFile("/favicon.ico", "./resources/favicon.ico") // 匯入所有模板,多級目錄結構需要這樣寫 router.LoadHTMLGlob("website/tpl/*/*") // 也可以根據handler,實時匯入模板。 // website分組 v := router.Group("/") { v.GET("/index.html", handler.IndexHandler) v.GET("/add.html", handler.AddHandler) v.POST("/postme.html", handler.PostmeHandler) } // 中介軟體 golang的net/http設計的一大特點就是特別容易構建中介軟體。 // gin也提供了類似的中介軟體。需要注意的是中介軟體只對註冊過的路由函數起作用。 // 對於分組路由,巢狀使用中介軟體,可以限定中介軟體的作用範圍。 // 大致分為全域性中介軟體,單個路由中介軟體和群組中介軟體。 // 使用全域性CORS中介軟體。 // router.Use(Cors()) // 即使是全域性中介軟體,在use前的程式碼不受影響 // 也可在handler中區域性使用,見api.GetUser //rate-limit 中介軟體 lmt := tollbooth.NewLimiter(1, nil) lmt.SetMessage("服務繁忙,請稍後再試...") // API分組(RESTFULL)以及版本控制 v1 := router.Group("/v1") { // 下面是群組中間的用法 // v1.Use(Cors()) // 單個中介軟體的用法 // v1.GET("/user/:id/*action",Cors(), api.GetUser) // rate-limit v1.GET("/user/:id/*action", LimitHandler(lmt), api.GetUser) //v1.GET("/user/:id/*action", Cors(), api.GetUser) // AJAX OPTIONS ,下面是有關OPTIONS用法的範例 // v1.OPTIONS("/users", OptionsUser) // POST // v1.OPTIONS("/users/:id", OptionsUser) // PUT, DELETE } srv := &http.Server{ Addr: ":80", Handler: router, ReadTimeout: 30 * time.Second, WriteTimeout: 30 * time.Second, } go func() { if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed { log.Fatalf("listen: %sn", err) } }() // 優雅Shutdown(或重新啟動)服務 // 5秒後優雅Shutdown服務 quit := make(chan os.Signal) signal.Notify(quit, os.Interrupt) //syscall.SIGKILL <-quit log.Println("Shutdown Server ...") ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) defer cancel() if err := srv.Shutdown(ctx); err != nil { log.Fatal("Server Shutdown:", err) } select { case <-ctx.Done(): } log.Println("Server exiting") }