【Golang】Gin框架路由转发处理模式

如果全都在main.go中处理请求,会导致文件过于混乱且开发效率低下。下面介绍如何将路由分组,并将请求处理代码放在同一文件夹下的不同的文件中,方便统一管理。

改进前

改进前的主程序大概是这样的,所有路由处理函数都在func main()里,十分臃肿。

package main

import (
	"encoding/xml"
	"fmt"
	"github.com/gin-gonic/gin"
	"net/http"
)

type LoginData struct {
	User     string `form:"user" xml:"user" binding:"required"`
	Password string `form:"password" xml:"password" binding:"required"`
}

func main() {
	router := gin.Default()

	router.POST("/post/xml", func(c *gin.Context) {
		xmlSliceData, err := c.GetRawData()
		if err != nil {
			c.JSON(http.StatusBadRequest, gin.H{
				"code": http.StatusBadRequest,
				"msg":  err.Error(),
			})
		}
		data := LoginData{}
		if err := xml.Unmarshal(xmlSliceData, &data); err == nil {
			fmt.Printf("%#v\n", data)
			if data.User == "root" && data.Password == "123456" {
				c.JSON(200, gin.H{"status": "you are logged in"})
			} else {
				c.JSON(401, gin.H{"status": "unauthorized"})
			}
		} else {
			c.JSON(http.StatusBadRequest, gin.H{
				"code": http.StatusBadRequest,
				"msg":  err.Error(),
			})
		}
    
	})
    
    router.POST("/post/baidu", func(c *gin.Context) {
		searchContent := c.PostForm("content")
		c.JSON(http.StatusOK, gin.H{
			"content": searchContent,
		})
	})
	err := router.Run(":8080")
	if err != nil {
		return
	}
}

改进后

创建路由的主程序,按照下面的方式,就能把主程序和各个组别的路由处理程序分离开来。注意导包路径。
在这里插入图片描述
可以看到这边使用了router包下的TestRouter函数,这个函数的作用是绑定路由,具体实现见后文的代码。

package main

import (
	"github.com/gin-gonic/gin"
	"learnGIN/router"
)

func main() {
	r := gin.Default()
	r.LoadHTMLGlob("views/*")
	// testRouter.go
	router.TestRouter(r)
	
	err := r.Run(":8080")
	if err != nil {
		return
	}
}

下面是包含路由信息的router文件,在TestRouter中,绑定了test组路由执行的函数,注意访问特定组路由的时候,需要加上组的路径。就下面的例子的login来说,需要访问http://localhost:8080/test/login

package router

import (
	"encoding/xml"
	"fmt"
	"github.com/gin-gonic/gin"
	"net/http"
)

type LoginData struct {
	User     string `form:"user" xml:"user" binding:"required"`
	Password string `form:"password" xml:"password" binding:"required"`
}

func TestRouter(router *gin.Engine) {

	testRouter := router.Group("/test")
	testRouter.GET("/login", func(c *gin.Context) {
		data := LoginData{}
		if c.ShouldBind(&data) == nil {
			if data.User == "root" && data.Password == "123456" {
				c.JSON(200, gin.H{"status": "you are logged in"})
			} else {
				c.JSON(401, gin.H{"status": "unauthorized"})
			}
		}
	})

	testRouter.GET("/html", func(c *gin.Context) {
		c.HTML(http.StatusOK, "baidu.html", gin.H{
			"title": "fake baidu",
		})
	})

	testRouter.POST("/baidu", func(c *gin.Context) {
		searchContent := c.PostForm("content")
		c.JSON(http.StatusOK, gin.H{
			"content": searchContent,
		})
	})

	testRouter.POST("/xml", func(c *gin.Context) {
		xmlSliceData, err := c.GetRawData()
		if err != nil {
			c.JSON(http.StatusBadRequest, gin.H{
				"code": http.StatusBadRequest,
				"msg":  err.Error(),
			})
		}
		data := LoginData{}
		if err := xml.Unmarshal(xmlSliceData, &data); err == nil {
			fmt.Printf("%#v\n", data)
			if data.User == "root" && data.Password == "123456" {
				c.JSON(200, gin.H{"status": "you are logged in"})
			} else {
				c.JSON(401, gin.H{"status": "unauthorized"})
			}
		} else {
			c.JSON(http.StatusBadRequest, gin.H{
				"code": http.StatusBadRequest,
				"msg":  err.Error(),
			})
		}

	})
}

控制器

在上一步的基础上,我们能够继续改进项目框架,形成我们现在主流的后端处理服务的模式。这边将router中的代码进一步分离,分成了router和controller两个部分。使router中的代码进一步精简,同时在controller控制器中,只需要专注于实现业务逻辑。
这是当前的项目架构。在这里插入图片描述

下面是router代码。

package router

import (
	"github.com/gin-gonic/gin"
	"learnGIN/myserver/controller"
)

func TestRouter(router *gin.Engine) {

	testRouter := router.Group("/test")
	testRouter.GET("/login", controller.TestController{}.Login)
	testRouter.GET("/html", controller.TestController{}.Html)
	testRouter.POST("/baidu", controller.TestController{}.Baidu)
	testRouter.POST("/xml", controller.TestController{}.Xml)
}

下面是控制器代码

package controller

import (
	"encoding/xml"
	"fmt"
	"github.com/gin-gonic/gin"
	"log"
	"net/http"
)

type TestController struct {
	BaseController
}

type LoginData struct {
	User     string `form:"user" xml:"user" binding:"required"`
	Password string `form:"password" xml:"password" binding:"required"`
}

func (t TestController) Login(c *gin.Context) {
	data := LoginData{}
	if err := c.ShouldBind(&data); err == nil {
		log.Printf("%#v\n", data)
		if data.User == "root" && data.Password == "123456" {
			t.Success(c, gin.H{"status": "you are logged in"})
		} else {
			t.Success(c, gin.H{"status": "unauthorized"})
		}
	} else {
		t.Fail(c, gin.H{"error": err.Error()})
	}
}

func (t TestController) Html(c *gin.Context) {
	c.HTML(http.StatusOK, "baidu.html", gin.H{
		"title": "fake baidu",
	})
}

func (t TestController) Baidu(c *gin.Context) {
	searchContent := c.PostForm("content")
	t.Success(c, gin.H{
		"content": searchContent,
	})
}

func (t TestController) Xml(c *gin.Context) {
	xmlSliceData, err := c.GetRawData()
	if err != nil {
		t.Fail(c, gin.H{
			"code": http.StatusBadRequest,
			"msg":  err.Error(),
		})
	}
	data := LoginData{}
	if err := xml.Unmarshal(xmlSliceData, &data); err == nil {
		fmt.Printf("%#v\n", data)
		if data.User == "root" && data.Password == "123456" {
			t.Success(c, gin.H{"status": "you are logged in"})
		} else {
			t.Success(c, gin.H{"status": "unauthorized"})
		}
	} else {
		t.Fail(c, gin.H{
			"code": http.StatusBadRequest,
			"msg":  err.Error(),
		})
	}
}

上面的controller继承自BaseController类,继承了Success和Fail函数。
下面是BaseController的代码。

package controller

import "github.com/gin-gonic/gin"

type BaseController struct{}

func (b *BaseController) Success(c *gin.Context, data gin.H) {
	c.JSON(200, data)
}

func (b *BaseController) Fail(c *gin.Context, data gin.H) {
	c.JSON(200, data)
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我可以为您提供一些关于使用 Golang Gin 框架进行项目实战的建议。 首先,您需要安装 GolangGin 框架。安装过程可以参考官方文档进行操作。然后,您可以按照以下步骤进行项目实战: 1. 创建项目结构 在您的 GOPATH 目录下创建一个新的项目文件夹。通常情况下,您需要为项目创建以下子文件夹: ``` . ├── main.go ├── config │ ├── config.go │ └── ... ├── controller │ ├── home_controller.go │ └── ... ├── middleware │ ├── auth.go │ └── ... ├── model │ ├── user.go │ └── ... ├── router │ ├── router.go │ └── ... └── service ├── user_service.go └── ... ``` 2. 创建由 在 router 文件夹中创建一个 router.go 文件,并编写以下代码: ```go package router import ( "github.com/gin-gonic/gin" "your-project/controller" "your-project/middleware" ) func SetupRouter() *gin.Engine { r := gin.Default() // 中间件 r.Use(middleware.AuthMiddleware()) // 由 r.GET("/", controller.HomePageHandler) return r } ``` 在上面的代码中,我们使用了 gin 框架中的 gin.Default() 函数来创建一个默认的 gin.Engine 实例,并使用中间件来进行身份验证。然后我们定义了一个由,将根径映射到 HomePageHandler 方法。 3. 创建控制器 在 controller 文件夹中创建 home_controller.go 文件,并编写以下代码: ```go package controller import ( "github.com/gin-gonic/gin" "your-project/service" ) func HomePageHandler(c *gin.Context) { users := service.GetAllUsers() c.JSON(200, users) } ``` 在上面的代码中,我们定义了一个 HomePageHandler 方法,它接收一个 gin.Context 对象作为参数,并使用 service 层中的 GetAllUsers 函数获取所有用户的数据。然后我们使用 JSON 方法将数据以 JSON 格式返回给客户端。 4. 创建服务层 在 service 文件夹中创建 user_service.go 文件,并编写以下代码: ```go package service import "your-project/model" func GetAllUsers() []*model.User { // TODO: 查询数据库获取所有用户数据 return []*model.User{} } ``` 在上面的代码中,我们定义了一个 GetAllUsers 函数,它返回一个包含所有用户数据的切片。在实际项目中,我们需要查询数据库来获取这些数据。 5. 创建模型 在 model 文件夹中创建 user.go 文件,并编写以下代码: ```go package model type User struct { ID int `json:"id"` Name string `json:"name"` Age int `json:"age"` } ``` 在上面的代码中,我们定义了一个 User 结构体,它包含三个字段:ID、Name 和 Age。这些字段将用于存储每个用户的信息。 6. 创建配置文件 在 config 文件夹中创建 config.go 文件,并编写以下代码: ```go package config const ( DB_HOST = "localhost" DB_PORT = "3306" DB_USERNAME = "root" DB_PASSWORD = "password" DB_NAME = "your-database-name" ) ``` 在上面的代码中,我们定义了一些常量,用于存储数据库配置信息。这些常量将在连接数据库时使用。 7. 连接数据库 在 service 层中的 GetAllUsers 函数中,我们需要连接数据库来获取所有用户数据。可以使用以下代码来连接 MySQL 数据库: ```go package service import ( "database/sql" "fmt" "your-project/config" "your-project/model" ) func GetAllUsers() []*model.User { // 连接数据库 db, err := sql.Open("mysql", fmt.Sprintf("%s:%s@tcp(%s:%s)/%s", config.DB_USERNAME, config.DB_PASSWORD, config.DB_HOST, config.DB_PORT, config.DB_NAME, )) if err != nil { panic(err) } defer db.Close() // 查询所有用户数据 rows, err := db.Query("SELECT * FROM users") if err != nil { panic(err) } // 解析查询结果 users := []*model.User{} for rows.Next() { user := &model.User{} err := rows.Scan(&user.ID, &user.Name, &user.Age) if err != nil { panic(err) } users = append(users, user) } return users } ``` 在上面的代码中,我们使用 sql.Open 函数来打开 MySQL 数据库连接。然后我们使用 db.Query 函数来查询所有用户数据,并使用 rows.Scan 函数将查询结果解析为 User 结构体的切片。 以上就是使用 Golang Gin 框架进行项目实战的流程。当然,在实际项目中,您需要根据自己的需求进行相应的修改和扩展。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值