golang aop

core.go

// @Title aop流程core
// @Description 参考gin库实现aop的core

package flow

import ctx "context"

type WrapF struct {
    f func(ctx ctx.Context) error

    ctx *Context
}

func WrapFunc(ctx ctx.Context, f func(ctx ctx.Context) error) *WrapF {
    return &WrapF{
        f:   f,
        ctx: newContext(ctx),
    }
}

func (wf *WrapF) UseBefore(f func(c *Context) error) {
    wf.ctx.addBeforeHandler(f)
}

func (wf *WrapF) UseBehind(f func(c *Context) error) {
    wf.ctx.addBehindHandler(f)
}

func (wf *WrapF) Handle() error {
    var err error
    wf.ctx.handlers = append(wf.ctx.handlers, wf.ctx.beforeHandlers...)
    wf.ctx.handlers = append(wf.ctx.handlers, func(c *Context) error {
        err := wf.f(wf.ctx)
        return err
    })
    wf.ctx.handlers = append(wf.ctx.handlers, wf.ctx.behindHandlers...)

    if len(wf.ctx.handlers) > 0 {
        err = wf.ctx.Next()
    }
    wf.ctx.Reset()

    return err
}

context.go

// @Title aop流程context
// @Description 参考gin库实现aop的context

package flow

import (
    ctx "context"
    "math"
)

const abort = math.MaxInt32 - 10000
const PAYLOADKEY = "payload"

type Context struct {
    ctx.Context
    offset         int
    beforeHandlers []func(ctx *Context) error
    behindHandlers []func(ctx *Context) error
    handlers       []func(ctx *Context) error
}

func newContext(ctx ctx.Context) *Context {
    return &Context{
        Context:        ctx,
        offset:         -1,
        beforeHandlers: make([]func(*Context) error, 0, 10),
        behindHandlers: make([]func(*Context) error, 0, 10),
        handlers:       make([]func(*Context) error, 0, 20),
    }
}

func (ctx *Context) Next() error {
    ctx.offset++
    s := len(ctx.handlers)
    for ; ctx.offset < s; ctx.offset++ {
        if !ctx.isAbort() {
            err := ctx.handlers[ctx.offset](ctx)
            if err != nil {
                return err
            }
        } else {
            return nil
        }
    }

    return nil
}

func (ctx *Context) Reset() {
    //ctx.PerRequestContext = &sync.Map{}
    ctx.offset = -1
    ctx.behindHandlers = ctx.handlers[:0]
    ctx.beforeHandlers = ctx.handlers[:0]
    ctx.handlers = ctx.handlers[:0]
}

func (ctx *Context) Abort() {
    ctx.offset = math.MaxInt32 - 10000
}

func (ctx *Context) isAbort() bool {
    if ctx.offset >= abort {
        return true
    }
    return false
}

func (ctx *Context) addBeforeHandler(f func(ctx *Context) error) {
    ctx.beforeHandlers = append(ctx.beforeHandlers, f)
}

func (ctx *Context) addBehindHandler(f func(ctx *Context) error) {
    ctx.behindHandlers = append(ctx.behindHandlers, f)
}

测试main.go

package main

import (
    "context"
    "errors"
    "fmt"
    "main/flow/flow"
)

type Payload struct {
    a int
    b int
}

type I interface {
    AlertTimeout()
}

func Fl() {
    payload := Payload{
        a: 1,
        b: 2,
    }
    a := 5
    c := context.WithValue(context.Background(), flow.PAYLOADKEY, &payload)
    wrapF := flow.WrapFunc(c, func(c context.Context) error {
        handle1(c, a)
        //pay := c.Value(flow.PAYLOADKEY).(*Payload)

        fmt.Println("000000000000000")

        return nil
    })
    wrapF.UseBefore(AlertTimeout)
    wrapF.UseBehind(Sentinel)
    err := wrapF.Handle()
    fmt.Println(err)

}

func main() {
    Fl()
    //wrapF2 := flow.WrapFunc(a, func() {
    //  handle2(a)
    //})
    //wrapF2.Use(AlertTimeout)
    //wrapF2.Use(Sentinel)
    //wrapF2.Handle()

}

func AlertTimeout(c *flow.Context) error {
    pay := c.Value(flow.PAYLOADKEY).(*Payload)
    pay.b = 1000

    fmt.Println("before")

    return nil
}

func Sentinel(c *flow.Context) error {
    fmt.Println("behind")

    return errors.New("ppppppppp")
}

func handle1(c context.Context, a int) int {
    fmt.Println("main1")
    //fmt.Println(c.Value(flow.PAYLOADKEY).(Payload).b)

    return 67
}

func handle2(a interface{}) {
    fmt.Println("main2")
}

请仔细阅读并实验该aop方案,代码参照gin、k8s中visitor、,包含对错误处理机制的调研

golang的错误处理机制

  1. 接口定义,直观返回error
package errs

import "github.com/pkg/errors"

type A interface {
    Enter() error
    Visit() error
    Leave() error
}

func B(a A, i int, j int) error {
    if err := a.Enter(); err != nil {

        return errors.WithMessage(err, "Enter failed")
    }

    if err := a.Visit(); err != nil {

        return errors.WithMessage(err, "visit failed")
    }

    if err := a.Leave(); err != nil {

        return errors.WithMessage(err, "leave failed")
    }

    return nil
}

分步骤处理,可以针对具体返回结果进行处理
2.屏蔽过程中的error处理

package errs

import "github.com/pkg/errors"

type A interface {
    Enter()
    Visit()
    Leave()
    
    Err() error
}

func B(a A) error {
    a.Enter()
    a.Visit()
    a.Leave()
    
    if err := a.Err(); err != nil {
        
        return errors.WithMessage(err, "failed")
    }

    return nil
}

type C struct {
    err error
}

func (l *C) Visit()  {
    if l.err != nil {
        return
    }
}

将error保存稻草对象内部,处理逻辑交给每个方法,本质上还是顺序执行
aop我第一版实现就是用这种方式处理错误
3.上面aop为第三种err处理,是目前最为优雅的,利用函数式编程的yan'c

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343

推荐阅读更多精彩内容

  • 原文链接:https://draveness.me/golang-101 Go 语言是一门简单、易学的编程语言,对...
    Draveness阅读 7,230评论 5 24
  • 关键点 希望通过下面的关键词,能够实现目的:能够快速的回忆,理解,复习 知识点: 1、Go语言中,运行时的错误处理...
    码二哥阅读 1,070评论 0 2
  • 写在前面 本文是Go语言的快速入门教程,适合于具有一定C语言或者Java语言基础的开发人员,如果您是一位Go...
    foundwei阅读 1,840评论 5 17
  • 在实际工程项目中,总是通过程序的错误信息快速定位问题,但是又不希望错误处理代码写的冗余而又啰嗦。Go语言没有提供像...
    drunkery阅读 840评论 0 0
  • 前言 最近在对极客时间毛剑老师的 Go 进阶训练营进行重温和学习汇总,这是一门比较偏向于工程化以及原理层面的的课程...
    pseudoyu阅读 557评论 0 0