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

业务描述

        两个接口(新增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添加到执行链中。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Golang是一种新兴的程序设计语言,被广泛应用于网络开发、云计算等领域。50k的高阶面试题对于不少求职者来说是一道难题,需要具备深厚的编程基础和实践经验,所以需要在以下几个方面下功夫: 1. 熟练掌握Golang的语法和特性,包括但不限于变量、函数、结构体、接口、包、并发等。透彻理解Golang的设计理念和哲学,掌握其优秀的并发性能和简洁的语法结构。 2. 掌握Golang的标准库和第三方库,并深入了解其内部实现和使用方法。掌握Golang的常用框架和工具,如Beego、Gin、Echo等,能够在实际工作中熟练使用。 3. 掌握Golang的性能优化和高可用架构设计,能够对应用程序进行优化,包括优化算法、内存管理、并发控制等方面。同时了解分布式系统、消息队列、容器化等技术,为应用程序提供高可用性和弹性。 4. 具备实际应用经验,能够解决实际工作中出现的各种问题。理解业务需求和用户体验,并能够为其提供创新的解决方案。 总之,要成为一名合格的Golang工程师,需要具备深厚的编程背景和实践经验。不断学习和深入掌握Golang的特性和应用场景,才能在高阶面试中获得成功。 ### 回答2: Golang 50K高阶面试题主要包括了Go语言的基础概念、并发编程、内存管理、网络等多个方面的考察。以下是对其中一些难点的简要解答。 并发编程: Go语言天生支持并发编程,因此并发编程是其核心优势之一。与其它编程语言不同,Go语言提供了一些内置的原语,如goroutine和channel,以便于编写并发代码。Goroutine是一个轻量级的线程,可与其他goroutine一起运行,而channel是用于在goroutine之间进行通信的并发安全的数据结构。通过这些特性,Go语言为开发并发程序提供了显着的便利。 内存管理: Go语言的内存管理是自动化的,开发者无需手动进行内存分配及释放。Go的垃圾回收机制可以自动检测不再使用的变量并将其回收,以便于再次利用。同时,Go语言还支持指针操作,但使用指针时需注意指针的生命周期以及指针空间的释放等问题,以免引起内存泄漏或者指针悬空等问题。 网络编程: Go语言提供了内置的网络库,能够轻松地进行TCP/IP和HTTP等协议的编程。与其它语言比较,Go语言提供了更高层次的API,使得开发网络应用变得更加简单,开发人员可以用更少更简洁的代码来实现同样的功能。 总的来说,Golang 50K高阶面试题主要考察了面试者在Go语言开发中的实际应用及其优势,特别是在并发编程、内存管理和网络编程等方面的应用。如果您希望在面试中表现更出色,需要熟练掌握Go语言的基本语法及其特性,并具有实际的项目经验。 ### 回答3: 首先,Golang 是一种高效、简洁、快速的编程语言,它在一定程度上成功地解决了传统语言的缺点,如并行编程的困难和内存泄漏等问题。而对于 Golang 高阶面试题,我们需要了解一些关键知识点。 首先是 Golang 的并发处理。Golang 的并发是基于 Goroutine 和 Channel 实现的,Goroutine 是轻量级线程,可以实现高度并发。而 Channel 是用来在 Goroutine 之间传递数据的通信机制。在面试中,常被问到如何确保 Goroutine 安全并实现并发。这时我们需要回答一些并发编程的关键问题,如共享资源、锁的使用和死锁避免等。 其次是 Golang 的内存管理。Golang 通过自动垃圾回收技术来实现内存管理和防止内存泄漏,而在面试中,常被问及如何避免内存泄漏,如何手动管理内存等问题,需要对 Golang 内存管理和垃圾回收机制有深入了解。 还有一些其他关键的问题,如异常处理和数据结构等,都需要我们进行深入学习和思考。在备战 Golang高阶面试时,除了掌握上述知识点外,还需要实践经验和多思考,不断完善自己的编程技能和能力。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值