golang使用高阶函数优化业务功能

2023-09-18 12:09:38

业务描述

        两个接口(新增Tag和更新Tag),在业务层均需要添加两个校验,校验Tag信息是否重复和Tag的数据中的编码是否重复。

基本实现

方式

        对应的增加两个校验的函数/方法,在接口实现中依次调用两个函数/方法进行校验。

优缺点

        实现简单;但重复代码多,后期再增加其他校验,扩展性较差。

高阶函数方式一

方式

        因为业务方法参数相同,业务层新增一个方法,包含一个函数类型参数(该函数即最终业务函数,新增/更新Tag),业务层新增方法:

func (t *TagUseCase) ValidateTag(ctx context.Context, req *Tag, f func(context.Context, *Tag) error) error {
   //校验一
   exists, err := t.repo.ValidateAppIdFieldExists(ctx, req.AppId, req.Field, req.IdString)
   if err != nil {
      return err
   }
   if exists {
      return errors.New(fmt.Sprintf("已存在:%v", req.Field))
   }
   //校验二
   flag, code := t.ValidateCodeUnique(req.Children)
   if flag {
      return errors.New(fmt.Sprintf("编码重复:%v", code))
   }
   //执行业务目标函数
   return f(ctx, req)
}

服务层调用:

func (s *TagService) CreateTag(ctx context.Context, req *pb.CreateTagRequest) (*pb.OperationTagReply, error) {
    //s.tuc.CreateTag为目标函数
	err := s.tuc.ValidateTag(ctx, tag, s.tuc.CreateTag)
	if err != nil {
		return nil, errors.New(0, err.Error(), "failed!")
	}
	return &pb.OperationTagReply{Msg: "success"}, nil
}

优缺点

        通过将目标函数参数化,将校验抽取到了一个方法中,后期如果增加其他校验,只需修改ValidateTag方法即可,有点类似于Java中的静态代理。重复代码很少,扩展性较好。但如果不同的业务需要的校验不完全相同,则存在问题。

高阶函数方式二

方式

        通过中间件实现(借鉴Kratos框架middleware),校验函数和目标函数的入参和返回值均相同(即使不同,可通过golang的闭包将函数包装成需要的函数签名),将其都作为一次处理,将所有的处理链接起来再执行。

定义中间件Handler类型:

定义中间件
type Handler func(ctx context.Context, req interface{}) error  #定义handler类型,即定义中间件和最终执行方法/函数的的声明
type Middleware func(Handler) Handler   # 定义Middle类型,即入参和返回值为相同类型的Handler,用于后面将其链接起来
# 将各中间件链接起来,next即最终要执行的函数,通过反向遍历的方式,将中间件按照添加的顺序依次链接,最先添加的在最外层,最先执行,结构类似:func(func(func(next)))
func Chain(m ...Middleware) Middleware {
   return func(next Handler) Handler {
      for i := len(m) - 1; i >= 0; i-- {
         next = m[i](next)
      }
      return next
   }
}

在业务层创建中间件:

func (t *TagUseCase) ValidatorTagExists() tagmiddleware.Middleware {
   return func(handler tagmiddleware.Handler) tagmiddleware.Handler {
      return func(ctx context.Context, req interface{}) (err error) {
         if v, ok := req.(*Tag); ok {
            exists, err := t.repo.ValidateAppIdFieldExists(ctx, v.AppId, v.Field, v.IdString)
            if err != nil {
               return err
            }
            if exists {
               return errors.New(fmt.Sprintf("该标签field已存在:%v", v.Field))
            }
         }
         return handler(ctx, req)
      }
   }
}

调用方式一:

#添加中间件:在业务层的结构体增加一个middleware字段,创建结构体时,将需要的中间件添加到该属性中
t.middleware = []tagmiddleware.Middleware{t.ValidatorTagExists(), t.ValidatorTagCodeRepeat()}

#提供给需要中间件的方法/函数调用: Chain首先获取到执行链,最终传入h进行调用
func (t *TagUseCase) Middleware(h tagmiddleware.Handler) tagmiddleware.Handler {
   return tagmiddleware.Chain(t.middleware...)(h)
}
最终调用:首先构建中间件,UpdateTag是最终要执行的方法。middleware(ctx,tag) 执行中间件和目标方法
middleware := s.tuc.Middleware(func(ctx context.Context, disposeReq interface{}) error {
   return s.tuc.UpdateTag(ctx, disposeReq.(*biz.Tag))
})
err = middleware(ctx, tag)

调用方式二:方式一提前指定了要执行那些中间件,不够灵活

#最终调用:在调用时指定要执行那些中间件。
middlewarex := tagmiddleware.Chain([]tagmiddleware.Middleware{s.tuc.ValidatorTagExists(), s.tuc.ValidatorTagCodeRepeat()}...)(func(ctx context.Context, disposeReq interface{}) error {
			return s.tuc.UpdateTag(ctx, disposeReq.(*biz.Tag))
		})
err = middlewarex(ctx, tag)

优缺点

        将校验和目标函数都作为handler处理,在调用时可以自定义设置要进行那些校验,灵活性高。但后期增加新的校验时,需要在多个调用的位置将新的校验handler添加到执行链中。

更多推荐

Android Jetpack解析之——LiveData

LiveData是一种可观察的数据存储器类。与常规的可观察类不同,LiveData具有生命周期感知能力,意指它遵循其他应用组件(如activity、fragment或service)的生命周期。这种感知能力可确保LiveData仅更新处于活跃生命周期状态的应用组件观察者。如果观察者(由Observer类表示)的生命周期

变量和配置文件

文章目录变量和配置文件1.变量1.1系统变量1.1.1系统变量分类1.1.2查看系统变量1.1.3修改系统变量的值1.2用户变量2.配置文件的使用2.1配置文件格式2.2启动命令与选项组2.3特定的MYSQL版本的专用选项组2.4同一个配置文件中多个组的优先级2.5命令行和配置文件中启动选项的区别变量和配置文件1.变量

ITR服务体系的常见问题和华为构建ITR的经验分享

大家好!前两天有一个企业负责客户服务、售后部门的朋友和华研荟探讨,企业的服务体系如何搭建,以及如何像华为一样构建ITR流程他的苦恼是,自己所带领的部门叫做客户服务中心,但是在公司内部不受重视,公司总觉得你们就是去现场安装、调试,出了问题去救个火,赶紧解决就好了,没什么难度嘛。在客户那边,每次出现问题总是一肚子抱怨,有的

Socks5代理、IP代理与网络安全:保护你的爬虫和隐私

在数字时代,数据是黄金,网络安全成为至关重要的问题。无论是保护个人隐私还是进行爬虫数据采集,代理技术已经成为网络工程师的必备工具。本文将深入探讨Socks5代理、IP代理以及它们在网络安全和爬虫应用中的重要性。1.了解Socks5代理Socks5代理是一种网络协议,允许应用程序通过中间服务器与目标服务器通信。与HTTP

分布式系统的 38 个知识点

天天说分布式分布式,那么我们是否知道什么是分布式,分布式会遇到什么问题,有哪些理论支撑,有哪些经典的应对方案,业界是如何设计并保证分布式系统的高可用呢?1.架构设计这一节将从一些经典的开源系统架构设计出发,来看一下,如何设计一个高质量的分布式系统;而一般的设计出发点,无外乎冗余:简单理解为找个备胎,现任挂掉之后,备胎顶

提高邮件营销效率,个性化推广利器——SerialMailer for Mac

在如今竞争激烈的市场中,个性化的营销和沟通对于吸引客户和保持关系至关重要。而SerialMailerforMac作为一款专业的邮件批量个性处理软件,能够帮助您轻松实现这一目标。SerialMailerforMac拥有强大且直观的界面,适用于任何规模的电子邮件营销活动。您可以轻松导入收件人列表,并根据收件人的个人信息、偏

【Linux】详解线程第一篇——由单线程到多线程的转变

线程详解前言正式开始啥是线程理解线程Windows和Linux下的线程Windows下的线程Linux下的线程对比重新理解进程理解曾经写的代码Linux的线程pthread库验证多线程在同一个进程中跑ps-aL线程资源线程切换成本低线程缺点线程异常线程等待pthread_create的第三个参数——回调函数的返回值终止

Flutter快速入门学习(一)

目录前言新建项目项目入口Dart的入口(项目的入口)布局视图组件Container(容器)Text(文本)Image(图片)Row(水平布局)和Column(垂直布局)ListView(列表视图)GridView(网格视图)Stack(层叠布局)Card(卡片)AppBar(应用栏)FloatingActionButt

Docker基本操作

目录Docker基本操作镜像操作拉取、查看镜像保存、导入镜像容器操作创建并运行一个容器进入容器,修改文件数据卷给nginx挂载数据卷给MySQL挂载本地目录Docker基本操作镜像操作拉取、查看镜像需求:从DockerHub中拉取一个nginx镜像并查看1)首先去镜像仓库搜索nginx镜像,比如DockerHub:2)

商城免费搭建之java商城 开源java电子商务Spring Cloud+Spring Boot+mybatis+MQ+VR全景+b2b2c

1.涉及平台平台管理、商家端(PC端、手机端)、买家平台(H5/公众号、小程序、APP端(IOS/Android)、微服务平台(业务服务)2.核心架构SpringCloud、SpringBoot、Mybatis、Redis3.前端框架VUE、Uniapp、Bootstrap/H5/CSS3、IOS、Android、小程

MySQL的索引概述

目录一、索引的描述二、如何在一个数据表中创建和删除索引呢?三、索引的"两面性"四、索引的适用场景一、索引的描述索引是数据库中一种用于提高数据检索速度和加快查询操作的数据结构。它类似于书籍的目录,可以快速定位到包含特定关键字或值的记录。索引的主要作用是加速数据库的数据检索过程,特别是在大型数据集和复杂查询的情况下。二、如

热文推荐