前言:

       什么是monkey ? 英语里面试猴子,但程序里面我们称之为 补丁.  用过python gevent的朋友应该很熟悉这个monkey补丁字眼吧。 对的,这里golang的monkey实现的功能跟gevent monkey是一样的,都是为了粗暴直接的替换标准库。


       我这里的场景很简单,就是想替换一些第三方库的json模块,像logrus就提供了优雅的jsonFormat接口,但其他的第三方库就没这么方便了,多少有些费劲的。 如果想批量替换标准库的一些方法,不外乎几种方法,一种是直接修改vendor源代码。,另一种是通过reflect和unsafe来实现monkey的逻辑。 这里离用的是bouk写的monkey库,github地址在这里  https://github.com/bouk/monkey


下面是我写的go monkey测试代码,  代码逻辑很简单,就是尝试用滴滴陶文的json-iterator替换encoding/json库。  这里多说一嘴,json-iterator在序列化struct的时候性能要比encoding/json高, map他俩半斤八两的。 json-iterator内部大量的使用sync.pool,一定程度上减少了对象的malloc创建。 还使用了缓存reflect技术,避免每次都去valueOf, typeOf …   

//xiaorui.cc
package main

import (
    "encoding/json"
    "fmt"

    "github.com/bouk/monkey"
    jjson "github.com/json-iterator/go"
)

type Dto struct {
    Name string `json:"name"`
    Addr string `json:"addr"`
    Like string `json:"like"`
}

func main() {
    monkey.Patch(json.Marshal, func(v interface{}) ([]byte, error) {
        fmt.Println("use jsoniter")
        return jjson.Marshal(v)
    })

    monkey.Patch(json.Unmarshal, func(data []byte, v interface{}) error {
        fmt.Println("use jsoniter")
        return jjson.Unmarshal(data, v)
    })
    dd := &Dto{
        Name: "xiaorui",
        Addr: "rfyiamcool@163.com",
        Like: "Python & Golang",
    }

    resDto := &Dto{}

    v, err := json.Marshal(dd)
    fmt.Println(string(v), err)

    errDe := json.Unmarshal(v, resDto)
    fmt.Println(resDto, errDe)

    fmt.Println("test end") 
}


下面是monkey patch的实现的主要逻辑.

func Patch(target, replacement interface{}) *PatchGuard {
	t := reflect.ValueOf(target)
	r := reflect.ValueOf(replacement)
	patchValue(t, r)

	return &PatchGuard{t, r}
}

func patchValue(target, replacement reflect.Value) {
	lock.Lock()
	defer lock.Unlock()

	if target.Kind() != reflect.Func {
		panic("target has to be a Func")
	}

	if replacement.Kind() != reflect.Func {
		panic("replacement has to be a Func")
	}

	if target.Type() != replacement.Type() {
		panic(fmt.Sprintf("target and replacement have to have the same type %s != %s", target.Type(), replacement.Type()))
	}

	if patch, ok := patches[target]; ok {
		unpatch(target, patch)
	}

	bytes := replaceFunction(*(*uintptr)(getPtr(target)), uintptr(getPtr(replacement)))
	patches[target] = patch{bytes, &replacement}
}


总结:

    monkey相关的逻辑放在main方法里就可以了,整个runtime时期只需要替换一次就ok了。golang monkey的方法虽然可以替换标准库和第三方库,但不管怎么说,还是显得有些粗暴了。还有就是 打入monkey后,pprof 和trace 显得有些异常,但业务没有影响。

   最后说一点,大家可能因为性能问题更换第三方库的时候,请一定要做好go test benchmark,压力测试的逻辑尽量贴合自己的业务。先前我们就因为golang nuts或者是社区的推荐,直接更换的库,但事实说明这不是很靠谱。 通过一次次的pprof分析,一次次的打脸。


END.