栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > 后端开发 > Java

仿照java的jdk动态代理实现go语言动态代理

Java 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

仿照java的jdk动态代理实现go语言动态代理

仿照java的jdk动态代理实现go语言动态代理

通过学习java的jdk动态代理和Cglib动态代理,仿照jdk动态代理用go实现了一个简单的动态代理

结构型模式 代理模式

代理模式中分为静态代理和动态代理。静态代理需要在编译前就要写好,而动态代理需要在运行时通过反射来实现方法增强。

静态代理:
  1. 代理类实现和目标类相同的接口,每个类都单独编辑一个代理类。
  2. 我们需要在代理类中,将目标类中的所有方法都要重新实现,并且为每个方法都附加相似的代码逻辑。
  3. 如果要添加方法增强的类不止一个,我们需要对每个类都创建一个代理类。
动态代理:
  1. 不需要为每个目标类编辑代理类。
  2. 在程序运行时,系统会动态地创建代理类,然后用代理类替换掉原始类。
  3. 一般采用反射实现。
代理模式的优点:
  1. 代理模式能将代理对象与真实被调用目标对象分离。
  2. 在一定程度上降低了系统的耦合性,拓展性好。
  3. 可以起到保护目标对象的作用。
  4. 可以增强目标对象的功能。
动态代理实现
package pro

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

//提供动态调用方法接口
type InvocationHandler interface {
   Invoke(proxy *Proxy, method *Method, args []interface{}) ([]interface{}, error)
}

//代理,用来总管代理类的生成
type Proxy struct {
   target  interface{}        //目标类,后面的类型和java的Object一样
   methods map[string]*Method //map用来装载待增强的不同的方法
   handle  InvocationHandler  //用来暴露统一invoke接口,类似多态
}

//创建新的代理
func NewProxy(target interface{}, h InvocationHandler) *Proxy {
   typ := reflect.TypeOf(target)          //用来显示目标类动态的真实类型
   value := reflect.ValueOf(target)       //获取目标类的值
   methods := make(map[string]*Method, 0) //初始化目标类的方法map
   //将目标类的方法逐个装载
   for i := 0; i < value.NumMethod(); i++ {
      method := value.Method(i)
      methods[typ.Method(i).Name] = &Method{value: method}
   }
   return &Proxy{target: target, methods: methods, handle: h}
}

//代理调用代理方法
func (p *Proxy) InvokeMethod(name string, args ...interface{}) ([]interface{}, error) {
   return p.handle.Invoke(p, p.methods[name], args)
}

//用来承载目标类的方法定位和调用
type Method struct {
   value reflect.Value //用来装载方法实例
}

//这里相当于调用原方法,在该方法外可以做方法增强,需要调用者自己实现!!!
func (m *Method) Invoke(args ...interface{}) (res []interface{}, err error) {
   defer func() {
      //用来捕捉异常
      if p := recover(); p != nil {
         err = errors.New(fmt.Sprintf("%s", p))
      }
   }()

   //处理参数
   params := make([]reflect.Value, 0)
   if args != nil {
      for i := 0; i < len(args); i++ {
         params = append(params, reflect.ValueOf(args[i]))
      }
   }

   //调用方法
   call := m.value.Call(params)

   //接收返回值
   res = make([]interface{}, 0)
   if call != nil && len(call) > 0 {
      for i := 0; i < len(call); i++ {
         res = append(res, call[i].Interface())
      }
   }
   return
}
测试
package pro

import (
   "fmt"
   "testing"
   "time"
)

func TestName(t *testing.T) {
   //这里对活动时长做统计
   people := &People{}   //创建目标类
   h := new(PeopleProxy) //创建接口实现类
   proxy := NewProxy(people, h)
   //调用方法
   ret, err := proxy.InvokeMethod("Work", "敲代码", "学习")
   if err != nil {
      fmt.Println(err)
   }

   fmt.Println(ret)
}

//目标类
type People struct {
}

func (p *People) Work(content string, next string) string {
   fmt.Println("活动内容是:" + content + ",接下来需要做:" + next)
   return "all right"
}

//用户需要自己实现的增强内容,需要实现InvocationHandler接口
type PeopleProxy struct {
}

//在这里做方法增强
func (p *PeopleProxy) Invoke(proxy *Proxy, method *Method, args []interface{}) ([]interface{}, error) {
   start := time.Now()
   defer fmt.Printf("耗时:%vn", time.Since(start))
   fmt.Println("before method")
   invoke, err := method.Invoke(args...)
   fmt.Println("after method")
   return invoke, err
}
输出

转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/1036257.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号