近期在公司實(shí)習(xí),參與了公司的一個(gè)分布式的應(yīng)用服務(wù)系統(tǒng)孵睬。系統(tǒng)采用Golang語言作為系統(tǒng)的開發(fā)語言播歼,在開發(fā)過程中采用了Go語言的反射函數(shù)的特性來取代了以前常使用的switch語法。
switch-case是一種多種選擇的語法掰读,其本質(zhì)與if-else方法差不多秘狞,都是通過判斷條件來執(zhí)行不同的方法。而Go提供了一種機(jī)制在運(yùn)行時(shí)更新變量和檢查它們的值蹈集,調(diào)用它們的方法烁试,和它們支持的內(nèi)在操作,但是在編譯時(shí)并不知道這些變量的類型拢肆。這種機(jī)制被稱為反射减响,反射也可以讓我們將類型本身作為第一類的值類型處理。
Web應(yīng)用路由問題
在我們編寫Web應(yīng)用過程中郭怪,常常會(huì)遇到一個(gè)路由需要對(duì)應(yīng)一個(gè)方法支示,我們會(huì)選擇使用switch的方法來進(jìn)行路由的匹配,若是路由匹配成功鄙才,我們會(huì)調(diào)用一個(gè)方法颂鸿,這種方法能夠很簡便的完成我們的工作,也便于程序員在編寫代碼過程中厘清思路攒庵。
問題:
在一個(gè)URL的路由中据途,我們?cè)趓equest中通過cmd的參數(shù)來對(duì)應(yīng)一個(gè)方法,這樣我們要如何根據(jù)一個(gè)cmd對(duì)應(yīng)一個(gè)方法叙甸?
可能針對(duì)這個(gè)問題有人會(huì)說,我們?yōu)槭裁床话裞md放到URL里面位衩,這樣的話就是一個(gè)方法對(duì)應(yīng)一個(gè)路由裆蒸,而且大多數(shù)的Web框架會(huì)通過回調(diào)函數(shù)來進(jìn)行函數(shù)調(diào)用,針對(duì)于這個(gè)問題糖驴,我只能說大部分的都是將cmd放到http的request里面的僚祷,具體的好處可能就是減少了對(duì)API的監(jiān)管佛致,以及當(dāng)路由比較多的時(shí)間能夠減少麻煩吧。
用switch方法實(shí)現(xiàn):
cmd := this.GetString("cmd")
switch cmd{
case "ls":
ls()
case "cd":
cd()
default:
fmt.Println("cmd method missing")
}
上述的方法先通過URL的參數(shù)獲取到cmd辙谜,然后通過cmd來調(diào)用相對(duì)應(yīng)的方法俺榆。在傳統(tǒng)的MVC的設(shè)計(jì)模式中,需要在Controller中添加switch方法装哆,同時(shí)需要在Model中實(shí)現(xiàn)相對(duì)應(yīng)的方法罐脊,總計(jì)修改了2個(gè)文件。
go語言反射函數(shù)
reflect包
在reflect包中蜕琴,主要通過Typeof()和Valueof()兩個(gè)方法來實(shí)現(xiàn)反射萍桌。兩個(gè)方法相互結(jié)合,能夠反射出被反射函數(shù)的全部信息凌简。
package main
import (
"fmt"
"reflect"
)
type Ref struct {
id int
name string
}
func (ref *Ref)GetName(){
fmt.Println("getName()函數(shù)")
}
func (ref *Ref)GetNameById(){
fmt.Println("getNameById()函數(shù)")
}
func main(){
t := reflect.TypeOf(&Ref{})
v := reflect.ValueOf(new(Ref))
fmt.Println(t)
fmt.Println(v)
for i:= 0; i< t.NumMethod();i++{
fmt.Println(t.Method(i).Name)
v.Method(i).Call(nil)
}
}
TypeOf()
TypeOf()函數(shù)主要是打印出被反射函數(shù)的類型上炎,其返回結(jié)果是reflect.Type類型。
在上面的示例中雏搂,通過Method().Name能夠反射其方法的函數(shù)名藕施。
常用的方法:
- func (t *rtype)String() string
- func (t *rtype)Name() string
- func (t *rtype)Kind() reflect.kind
- func (t *rtype)Method(int) reflect.Method
- func (t *rtype)Elem() reflect.Type
- func (t *rtype)In(int) reflect.Type
ValueOf()
ValueOf()函數(shù)主要是打印出被反射函數(shù)的類型,其返回結(jié)果是reflect.Value類型凸郑。
在上面的示例中裳食,通過Method().Call()能夠反射出其函數(shù)并執(zhí)行。
常用的方法:
- func (v Value)String() string
- func (v Value)Elem() reflect.Value
- func (v Value)Method(int) reflect.Value
- func (v Value)Call(in []Value) (r []Value)
反射的實(shí)現(xiàn)過程
由于有反射的存在线椰,因此在傳統(tǒng)的MVC的設(shè)計(jì)模式中胞谈,當(dāng)我們添加服務(wù)時(shí),不需要修改Controller端的代碼憨愉,Controller只需要維持一個(gè)map的表烦绳,里面的就來存儲(chǔ)需要被反射的models。
package server
import(
"reflect"
"fmt"
)
// ReServer 來保存map的結(jié)構(gòu)體
type ReServer struct {
m map[string]interface{}
}
// RegisterService 注冊(cè)服務(wù)
func (this *ReServer)RegisterService(service interface{})(err error){
serviceType := reflect.TypeOf(service).Elem()
ServiceName := serviceType.Name()
if _,ok := this.m[ServiceName]
if ok {
fmt.Println("service has been registered")
}else{
this.m[ServiceName] = service
}
return
}
// Start 服務(wù)啟動(dòng)
func (this *ReServer)Start(){
for k,v := range this.m {
// 里面根據(jù)業(yè)務(wù)邏輯執(zhí)行想要的方法
}
}
在上訴的例子中配紫,通過對(duì)services的服務(wù)注冊(cè)径密,就能夠通過Start()函數(shù)發(fā)現(xiàn)服務(wù),并且根據(jù)業(yè)務(wù)來實(shí)現(xiàn)自己的代碼躺孝。
package main
import (
"server"
)
type Server struct {
}
func (server *Server)funOne(){
fmt.Println("Server FunOne")
}
func main(){
reServer := &server.ReServer{
m: make(map[string]interface{})
}
err := reServer.RegisterService(new(Server))
reServer.Start()
}
因此在我們主函數(shù)中享扔,導(dǎo)入封裝好的包,只需要注冊(cè)一個(gè)結(jié)構(gòu)體植袍,就能夠?qū)⒆约旱姆椒ǚ瓷涑鰜韺?shí)現(xiàn)惧眠。
對(duì)應(yīng)上面的Web的路由問題,我們將Controller進(jìn)行封裝于个,然后將Model進(jìn)行反射氛魁,當(dāng)我們業(yè)務(wù)增加時(shí),我們?cè)贛odel里面添加就可以了,不需要修改Controller秀存。