我爱编程

Golang 反射实现依赖注入

2017-12-25  本文已影响178人  sipt

Golang 反射实现依赖注入

Coding/Golang #Golang #Golang/reflect

依赖注入

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

构想

结构体改造前的代码

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)
                }
            }
        }
    }
}
上一篇下一篇

猜你喜欢

热点阅读