想入门go语言的web开发,找到好几款框架gin、iris、echo。
请问go语言web框架gin与iris及echo如何选择?
网友回复
前言 由于golang提供了完善的net/http标准库,基于该标准库实现一个web框架的难度相比其他语言低了不少,所以go web框架简直就是百花齐放。从老牌的revel和beego,到新出的gin,和iris等,而且还有一些类似于chi这种router。个人一般小项目,尤其是中间件需要暴露一些http接口的,基本就使用chi即可。 本次测试主要是gin iris echo 这三个框架。侧重在于高性能,从并发和json序列化和反序列化两个方面来测评,毕竟后台项目侧重的也就是这两个方面。 测试 测试环境说明 为了选择符合重IO的框架,现设定如下场景的demo,demo的具体要求如下: 打开日志功能(模拟正常业务时也会记录日志),在请求开始和结束时分别记录一条日志 接口中用sleep暂停1秒,假设这里的网络IO操作(同时更容易从日志看出是否协程并发的行为) 用POST接口做测试,接口中不进行任何处理,把接收到的body直接序列化返回(序列化和反序列化是框架最高频的动作) 打开框架的accesslog功能 测试工具以及场景如下 测试工具使用经典的jmeter,直接使用GUI界面测试 场景分为10线程并发,100线程并发,500线程并发,1000线程并发和1500线程并发 所有结果都只看jmeter的聚合报告,重点查看吞吐量、时间和错误数 所有demo启动的时候均启动单线程,异步框架不限制协程的数量,设置GOMAXPROCS=1 所有测试均在本地,压测时长两分钟 测试时采用POST请求,数据样本有565bytes、5KB、10KB、50KB和100KB,每个样本都要在不同并发线程上测试 测试代码 gin:
package main import ( "log" "net/http" "time" "github.com/gin-gonic/gin" ) // Agent ... type Agent struct { AgentID string `json:"agent_id"` QueuedAt string `json:"queued_at"` QueuedBy string `json:"queued_by"` } // Details ... type Details struct { EventID string `json:"event_id"` Endpoint string Metric string Content string Priority int Status string } // Test1 ... type Test1 struct { Agent Agent Details Details Description string EventType string `json:"event_type"` ServiceKey string `json:"service_key"` } // Test2 test2 type Test2 struct { Data []*Test1 } func main() { r := gin.New() // Global middleware // Logger middleware will write the logs to gin.DefaultWriter even if you set with GIN_MODE=release. // By default gin.DefaultWriter = os.Stdout r.Use(gin.Logger()) r.GET("/ping", func(c *gin.Context) { c.JSON(200, gin.H{ "message": "pong", }) }) r.POST("/v1/test", func(c *gin.Context) { var test Test1 if err := c.BindJSON(&test); err == nil { log.Println("========================start io=====================") time.Sleep(time.Duration(1) * time.Second) log.Println("=========================end io=======================") c.JSON(http.StatusOK, test) } else { c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) } }) r.POST("/v2/test", func(c *gin.Context) { var test Test2 if err := c.BindJSON(&test); err == nil { log.Println("========================start io=====================") time.Sleep(time.Duration(1) * time.Second) log.Println("=========================end io=======================") c.JSON(http.StatusOK, test) } else { c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) } }) r.POST("/v3/test", func(c *gin.Context) { var test Test2 if err := c.BindJSON(&test); err == nil { log.Println("========================start io=====================") time.Sleep(time.Duration(1) * time.Second) log.Println("=========================end io=======================") c.JSON(http.StatusOK, test) } else { c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()}) } }) r.POST("/v4/test", func(c *gin.Context) { var test Test2 if err := c.BindJSON(&test); err == nil { log.Println("========================start io=====================") time.Sleep(time.Duration(1) * time.Second) log.Println("=========================end io===========...
点击查看剩余70%