当前位置: 首页 > news >正文

go-zero中统一返回前端数据格式的几种方式

方式一、直接定义一个成功和失败的方法,在代码里面修改(对代码有侵入,每次都要修改代码)
  • 1、封装一个统一返回的方法

    package utilsimport ("github.com/zeromicro/go-zero/rest/httpx""net/http"
    )type Body struct {Code    int         `json:"code"`Message string      `json:"message"`Result  interface{} `json:"result,omitempty"`
    }func Response(w http.ResponseWriter, code int, message string, data interface{}) {httpx.OkJson(w, Body{Code:    code,Message: message,Result:  data,})
    }// Success 成功的请求
    func Success(w http.ResponseWriter, data interface{}) {Response(w, 0, "请求成功", data)
    }// Fail 失败的请求
    func Fail(w http.ResponseWriter, message string) {Response(w, 1, message, nil)
    }
  • 2、修改代码的返回值方法调用

    func GetTestHandler(svcCtx *svc.ServiceContext) http.HandlerFunc {return func(w http.ResponseWriter, r *http.Request) {l := demo.NewGetTestLogic(r.Context(), svcCtx)resp, err := l.GetTest()if err != nil {//httpx.ErrorCtx(r.Context(), w, err)utils.Fail(w, err.Error())} else {//httpx.OkJsonCtx(r.Context(), w, resp)utils.Success(w, resp)}}
    }
    
方式二、在api文件定义的时候直接返回的数据结构(在定义api的时候繁琐了)
  • 1、api文件中定义

    type Response {Code string      `json:"code"`Msg  string      `json:"msg"`Data interface{} `json:"data"`
    }@server (prefix: demo/v1group:  demo
    )
    service demo-api {@doc "测试"@handler GetTestget /testApi returns (Response)}
    
  • 2、在每次操作的时候可以写上

    func (l *GetTestLogic) GetTest() (resp *types.Response, err error) {return &types.Response{Code: "0",Msg:  "成功",Data: "hello world",}, nil
    }
    
  • 3、参考官网地址

方式三、自定义模版,直接修改返回的数据方法(可以接受)
  • 1、定义方法

    package utilsimport ("encoding/json""encoding/xml""errors""github.com/zeromicro/go-zero/core/logx""github.com/zeromicro/go-zero/rest/httpx""net/http"
    )type Body struct {Code    int         `json:"code"`Message string      `json:"message"`Result  interface{} `json:"result"` // omitempty json里面加了这个表示如果为空的时候字段不返回
    }type PageVo struct {Data       interface{} `json:"data"`       // 数据Total      int64       `json:"total"`      // 总条数PageSize   int64       `json:"pageSize"`   // 当前条数PageNumber int64       `json:"pageNumber"` // 当前页数
    }// Response 一般返回
    func Response(w http.ResponseWriter, resp interface{}, err error) {if err != nil {httpx.OkJson(w, Body{Code:    1,Message: err.Error(),Result:  nil,})} else {httpx.OkJson(w, Body{Code:    0,Message: "请求成功",Result:  resp,})}
    }// BuildPageData 返回分页数据
    func BuildPageData(w http.ResponseWriter, resp interface{}, total, pageSize, pageNumber int64, err error) {if err != nil {httpx.OkJson(w, Body{Code:    1,Message: err.Error(),Result: PageVo{Data:       nil,        // 数据Total:      0,          // 总条数PageSize:   pageSize,   // 当前条数PageNumber: pageNumber, // 当前页数},})} else {httpx.OkJson(w, Body{Code:    0,Message: "请求成功",Result: PageVo{Data:       resp,       // 数据Total:      total,      // 总条数PageSize:   pageSize,   // 当前条数PageNumber: pageNumber, // 当前页数},})}
    }func WriteString(w http.ResponseWriter, code int, v string) {w.Header().Set("Content-Type", "text/plain; charset=utf-8")w.WriteHeader(code)if n, err := w.Write([]byte(v)); err != nil {// http.ErrHandlerTimeout has been handled by http.TimeoutHandler,// so it's ignored here.if !errors.Is(err, http.ErrHandlerTimeout) {logx.Errorf("write response failed, error: %s", err)}} else if n < len(v) {logx.Errorf("actual bytes: %d, written bytes: %d", len(v), n)}
    }func WriteXml(w http.ResponseWriter, code int, v interface{}) {bs, err := xml.Marshal(v)if err != nil {http.Error(w, err.Error(), http.StatusInternalServerError)return}w.Header().Set("Content-Type", "text/xml")w.Header().Set("response_body", string(bs))w.WriteHeader(code)if n, err := w.Write(bs); err != nil {// http.ErrHandlerTimeout has been handled by http.TimeoutHandler,// so it's ignored here.if !errors.Is(err, http.ErrHandlerTimeout) {logx.Errorf("write response failed, error: %s", err)}} else if n < len(bs) {logx.Errorf("actual bytes: %d, written bytes: %d", len(bs), n)}
    }func WriteFile(w http.ResponseWriter, fileBytes []byte, err error) {var (b []byte)if err != nil {httpx.OkJson(w, err)b, _ = json.Marshal(err)w.Header().Set("response_body", string(b))return}w.Header().Set("Content-Type", "application/octet-stream")if n, err := w.Write(fileBytes); err != nil {// http.ErrHandlerTimeout has been handled by http.TimeoutHandler,// so it's ignored here.if !errors.Is(err, http.ErrHandlerTimeout) {logx.Errorf("write response failed, error: %s", err)}} else if n < len(fileBytes) {logx.Errorf("actual bytes: %d, written bytes: %d", len(fileBytes), n)}w.WriteHeader(200)
    }
  • 2、定义模版handler.tpl

    package {{.PkgName}}import ("net/http""github.com/zeromicro/go-zero/rest/httpx"{{.ImportPackages}}
    )func {{.HandlerName}}(svcCtx *svc.ServiceContext) http.HandlerFunc {return func(w http.ResponseWriter, r *http.Request) {{{if .HasRequest}}var req types.{{.RequestType}}if err := httpx.Parse(r, &req); err != nil {utils.TranslatorError(w, r, err)return}{{end}}l := {{.LogicName}}.New{{.LogicType}}(r.Context(), svcCtx){{if .HasResp}}resp, {{end}}err := l.{{.Call}}({{if .HasRequest}}&req{{end}}){{if .HasResp}}utils.Response(w, resp, err){{else}}utils.Response(w, nil, err){{end}}}
    }
  • 3、使用自己的模版生成方法

    goctl api go -api *api --dir . --style=goZero --home ../../goctl
    
  • 4、生成的代码中

    func DemoHandler(svcCtx *svc.ServiceContext) http.HandlerFunc {return func(w http.ResponseWriter, r *http.Request) {var req types.Requestif err := httpx.Parse(r, &req); err != nil {httpx.ErrorCtx(r.Context(), w, err)return}l := demo.NewDemoLogic(r.Context(), svcCtx)resp, err := l.Demo(&req)utils.Response(w, resp, err)}
    }
    
  • 5、这种方式不好的地方,在每个项目中导包可能不一样的,需要手动改动模版,或者把封装的方法封装成一个库发布

方式四、使用拦截器(最优方案)
  • 1、创建自己返回的方法

    package utilsimport ("context""github.com/zeromicro/go-zero/core/logx""net/http"
    )type Response struct {Code    int64       `json:"code"`Message string      `json:"message"`Result  interface{} `json:"result,omitempty"`
    }func Success(data interface{}) *Response {return &Response{Code:    0,Message: "请求成功",Result:  data,}
    }func Fail(err string) *Response {return &Response{Code:    1,Message: err,Result:  nil,}
    }func OkHandler(_ context.Context, v interface{}) any {return Success(v)
    }func ErrHandler(name string) func(ctx context.Context, err error) (int, any) {return func(ctx context.Context, err error) (int, any) {// 日志记录logx.WithContext(ctx).Errorf("【%s】 err %v", name, err)return http.StatusBadRequest, Fail(err.Error())}
    }
    
  • 2、在项目启动文件中添加自己的方法

在这里插入图片描述

// 使用拦截器
httpx.SetOkHandler(utils.OkHandler)
httpx.SetErrorHandlerCtx(utils.ErrHandler(c.Name))
http://www.lryc.cn/news/421239.html

相关文章:

  • 【向量数据库】Ubuntu编译安装FAISS
  • 制造知识普及(九)--企业内部物料编码(IPN)与制造商物料编码(MPN)
  • 【整数规划】+【0—1规划】解决优化类问题(Matlab代码)
  • Linux下如何使用Curl进行网络请求
  • PostgreSQL 触发器
  • LeetCode——3131.找出与数组相加的整数I
  • 【SpringMVC】详细了解SpringMVC中WEB-INF 目录资源,视图解析器和静态资源放行的使用。
  • 如何学好uni-app
  • C++ QT使用stackwidget实现页面切换(含源码)
  • 打工人上班适合用的蓝牙耳机推荐?几款开放式耳机推荐
  • 一款.NET开发的AI无损放大工具
  • 编程新手必看:彻底理解!与~的取反操作
  • 【LeetCode】54. 螺旋矩阵
  • 计算机毕业设计 奖学金评定管理系统 Java+SpringBoot+Vue 前后端分离 文档报告 代码讲解 安装调试
  • 【JavaWeb项目】——外卖订餐系统之商家添加餐品、修改餐品、查询热卖餐品、查询出售车、进行发货操作
  • 制作抖音私信卡片 - 一键调起并跳转微信二维码
  • 赋能未来园区:TSINGSEE视频AI智能管理平台如何引领园区管理智慧化转型
  • Linux逻辑卷管理LVM
  • 团队诊断工具TDS
  • DC-5靶机渗透测试
  • 16、电科院FTU检测标准学习笔记-基本性能2
  • MySQL——使用Python操作MySQL
  • Flink的DataStream状态管理
  • Daiqile SQL注入绕过
  • 用Python轻松移除PDF中的注释
  • 51单片机—串口
  • vue 通过 this.$refs 创建方法i向子组件传参让子组件更新
  • Java设计模式以及代理模式
  • Elasticsearch 索引库管理:查询、修改与删除
  • 视频大怎么压缩小?分享3种视频压缩方法