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)
}
相关推荐
疯笔码良1 小时前
【Flutter】flutter安装并在Xcode上应用
flutter·macos·xcode
寻星探路2 小时前
Java EE初阶启程记13---JUC(java.util.concurrent) 的常见类
java·开发语言·java-ee
哲Zheᗜe༘2 小时前
了解学习Python编程之python基础
开发语言·python·学习
youliroam3 小时前
成语接龙学习
学习·golang·uniapp·成语接龙
落日漫游3 小时前
数据结构笔试核心考点
java·开发语言·算法
寻找华年的锦瑟3 小时前
Qt-配置文件(INI/JSON/XML)
开发语言·qt
HY小海3 小时前
【C++】AVL树实现
开发语言·数据结构·c++
workflower3 小时前
Fundamentals of Architectural Styles and patterns
开发语言·算法·django·bug·结对编程
Roc-xb3 小时前
ModuleNotFoundError: No module named ‘conda_token‘
开发语言·python·conda
人工干智能4 小时前
Python 开发中:`.ipynb`(Jupyter Notebook 文件)和 `.py`(Python 脚本文件)
开发语言·python·jupyter