go 切面 AOP 实现

go AOP 实现

使用Go语言的反射机制和函数类型实现AOP,通过在需要切入的函数前后添加额外的逻辑代码实现AOP

go 复制代码
package main

import (
	"errors"
	"fmt"
	"log"
	"reflect"
)

// User 结构体表示一个用户
type User struct {
	ID   int
	Name string
}

type Aspect struct {
	Before []func([]reflect.Value) error
	After  []func([]reflect.Value) error
}

// Apply 利用反射执行切面的前置和后置处理函数
func (a *Aspect) Apply(targetFunc interface{}, args []reflect.Value) ([]reflect.Value, error) {
	for _, beforeFunc := range a.Before {
		if err := beforeFunc(args); err != nil {
			return nil, err
		}
	}

	result := reflect.ValueOf(targetFunc).Call(args)
	for _, v := range result {
		if v.Type().AssignableTo(reflect.TypeOf((*error)(nil)).Elem()) {	
			if v.IsNil() {
				continue
			}
			if err, ok := v.Interface().(error); ok {
				if err != nil {
					return nil, err
				}
			} else {  
				log.Panicln("Failed to convert return value to error")  
			}  
		}
	}

	for _, afterFunc := range a.After {
		go afterFunc(args)
	}

	return result, nil
}

func BeforeAspect1(args []reflect.Value) error {
	id := args[0].Interface().(int)
	log.Println("BeforeAspect1 get id:", id)
	return nil
}

func BeforeAspect2(args []reflect.Value) error {
	name := args[1].Interface().(string)
	log.Println("BeforeAspect2 get name:", name)
	return nil
}

func BeforeAspect3(args []reflect.Value) error {
	user := args[2].Interface().(*User)
	log.Println("BeforeAspect3 get user:", user.ID, user.Name)
	return nil
	return errors.New("BeforeAspect3 error")
}

func AfterAspect1(args []reflect.Value) error {
	log.Println("AfterAspect1")
	return nil
}

func TargetFunction(id int, name string, user *User) (*User, error) {
	user.ID = 13
	user.Name = "John Doe"
	fmt.Printf("Executing target function with ID: %d, Name: %s, User: %+v\n", id, name, user)
	return user, nil
}

func main() {
	myAspect := &Aspect{
		Before: []func([]reflect.Value) error{
			BeforeAspect1, 
			BeforeAspect2,
			BeforeAspect3,
		},
		After: []func([]reflect.Value) error{
			AfterAspect1,
		},
	}

	// 使用切面执行目标函数
	targetFuncValue := reflect.ValueOf(TargetFunction)

	// 构造目标函数的参数值
	id := 1
	name := "John"
	user := &User{ID: 100, Name: "Alice"}
	args := []reflect.Value{
		reflect.ValueOf(id),
		reflect.ValueOf(name),
		reflect.ValueOf(user),
	}

	result, err := myAspect.Apply(targetFuncValue.Interface(), args)
	if err != nil {
		fmt.Println("have err:", err)
		return
	}

	fmt.Println("Target function result:", result)
}
相关推荐
编码浪子3 分钟前
趣味学RUST基础篇(构建一个命令行程序2重构)
开发语言·重构·rust
echoarts40 分钟前
MATLAB R2025a安装配置及使用教程(超详细保姆级教程)
开发语言·其他·matlab
阿方.9181 小时前
《数据结构全解析:栈(数组实现)》
java·开发语言·数据结构
Dovis(誓平步青云)1 小时前
《探索C++11:现代语法的内存管理优化“性能指针”(下篇)》
开发语言·jvm·c++
charlie1145141911 小时前
前端三件套简单学习:HTML篇1
开发语言·前端·学习·html
kebeiovo2 小时前
项目必备流程图,类图,E-R图实例速通
开发语言·r语言·流程图
软件开发-NETKF88882 小时前
JSP到Tomcat特详细教程
java·开发语言·tomcat·jsp·项目运行
ftswsfb2 小时前
现代C++:现代C++?
开发语言·c++
乌萨奇也要立志学C++2 小时前
【C++详解】C++ 智能指针:使用场景、实现原理与内存泄漏防治
开发语言·c++
minji...2 小时前
C++ 详细讲解vector类
开发语言·c++