Golang 反射实现依赖注入

Golang 反射实现依赖注入

Coding/Golang #Golang #Golang/reflect

依赖注入

本人因为Java转入Golang就会带入一些Java的思维,依赖注入是个好东西,概念不多缀述。

构想

  • 创建包含要注入对象的实例,并存入一个Map
  • 创建需要被别的对象注入的对象实例,也存入这个Map
  • 对象注册完成后,调用Inject方法,Inject会遍历整个Map,找出对象中所有要注入的属性,找出并注入

结构体改造前的代码

func main(){
    runner := &Runner{}
    eater := &Eater{}
    people := NewPeople(runner, eater)
    people.Run.Run()
}
func NewPeople(run IRun, eat IEat) IPeople {
    return & People{
        Run: run,
        Eat: eat,
    }
}
type People struct {
    Run IRun
    Eat IEat
}

结构体改造后的代码

func main(){
    core.register("run", &Runner{}) //以run为名注册 Runner
    core.register("eat", &Eater{}) //以eat为名注册 Eater
    people := &People{}
    core.autoRegister(people) //以*main.People为名注册 People
    core.inject() //注入带有Tag:"auto"的所有对象
    people.Run.Run()
}
type People struct {
    Run IRun `auto:"run"`
    Eat IEat `auto:"eat"`
}

实现

Tag

使用Tag auto标记出需要自动注入的对象,如: auto:"run"取出以run名注册的对象并注入。
遍历字段找出所有Tag为auto的字段。

value := reflect.ValueOf(obj)
if value.Kind() == reflect.Ptr {
    value = value.Elem()
}
for i := 0; i < value.NumField(); i++ {
    name := value.Type().Field(i).Tag.Get("auto")
    // 获取到tag auto 的值
}

反射注入

在这里给一个Public属性修改值比较简单:

field := value.Field(i) // 获取字段
field.Set(value) // 设置值

但是当值是个Private时,调用Set会发生Panic,可以使用方法CanSet判断出来,主要问题还是如何为一个Private对象设置值:

field := value.Field(i)
field = reflect.NewAt(field.Type(), unsafe.Pointer(field.UnsafeAddr())).Elem()
field.Set(value)

这些都是标准库中的方法,没有存在很花哨的东西。

附录

import (
    "fmt"
    "reflect"
    "unsafe"
)

const (
    InjectorTag = "auto"
)

var objs map[string]reflect.Value

func init() {
    objs = make(map[string]reflect.Value, 10)
}

// Register 注册对象
func Register(name string, v interface{}) {
    objs[name] = reflect.ValueOf(v)
}

// AutoRegister 注册对象
func AutoRegister(v interface{}) {
    rv := reflect.ValueOf(v)
    Register(rv.Type().String(), rv)
}

// Get 获取注册对象
func Get(key string) interface{} {
    v, ok := objs[key]
    if ok {
        return v.Interface()
    }
    return nil
}

// Remove 删除注册对象
func Remove(key string) {
    delete(objs, key)
}

func Inject() {
    for _, v := range objs {
        value := v
        if value.Kind() == reflect.Ptr {
            value = value.Elem()
        }
        for i := 0; i < value.NumField(); i++ {
            name := value.Type().Field(i).Tag.Get(InjectorTag)
            temp, ok := objs[name]
            if ok {
                field := value.Field(i)
                if field.CanSet() {
                    field.Set(temp)
                } else {
                    field = reflect.NewAt(field.Type(), unsafe.Pointer(field.UnsafeAddr())).Elem()
                    field.Set(temp)
                }
            }
        }
    }
}
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,904评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,764评论 18 399
  • 一. Java基础部分.................................................
    wy_sure阅读 3,835评论 0 11
  • 此文为本人学习guice的过程中,翻译的官方文档,如有不对的地方,欢迎指出。另外还有一些附件说明、吐槽、疑问点,持...
    李眼镜阅读 3,532评论 2 5
  • 首先,跑步这个事情我并没有得坚持。所以我没资格写这次的周作业。以下内容完全只为奏字数而拼奏字数。 记得第一次参加自...
    爱吃鱼的小白羊阅读 188评论 2 0