第15课:GO语言"类"操作

当发现go语言没有类时,小伙伴都震惊了,但是如果你要保证好的代码结构与内部逻辑组织结构,类这种模式貌似又是不可或缺的
那么问题来了,go语言有没有办法来模拟类操作呢?答案是肯定的

首先我们来看go语言struct数据类型
type Godeye struct {  
    name string  
    age int  
}
初始化
p := Godeye{"godeye", 28}  //有序  
p := Godeye{age:28}        //无序

最有意思的是,struct可以增加关联函数
func funcName(varName1 typeName2[,varName2 typeName2, ...]) typeName {...}
例子
func (this Godeye) Test(area string) string {
	return *this.name + area
}
其中Godeye就是对应的Godeye struct而this.name就是Godeye中的name

有没有发现,和类何其的相似,在一个go文件里,可以定义一个struct,name age相当于类里的变量
Test相当于类里的方法

但是光模拟是没有用的,我们该怎么调用呢?
类调用很简单,直接用classname->Test()直接调用类对应的方法
go语言就不同,甚至可以说复杂多,首先你要先理解reflect反射的概念,其次还要熟悉interface,再次还要理解new make是怎么运作的

reflect反射:调用的关键
反射就是动态运行时的状态。对应的包是reflect包
i在这里是一个struct,转化为reflect对象
t := reflect.TypeOf(i)    //得到类型的元数据,通过t我们能获取类型定义里面的所有元素
v := reflect.ValueOf(i)   //得到实际的值,通过v我们获取存储在里面的值,还可以去改变值
转化为reflect对象之后我们就可以进行一些操作了,也就是将reflect对象转化成相应的值,例如
tag := t.Elem().Field(0).Tag        //获取定义在struct里面的标签
name := v.Elem().Field(0).String()  //获取存储在第一个字段里面的值

interface:函数的参数
简单的说,interface是一组method的组合,我们通过interface来定义对象的一组行为
我们可以通过定义interface参数,让函数接受各种类型的参数

还是用一个可运行的完整例子来说明:
1.新建一个main.go
package main

import (
	"./action"
	"fmt"
	"log"
	"net/http"
	"net/url"
	"reflect"
)

func main() {
	http.HandleFunc("/index", do)
	err := http.ListenAndServe(":8080", nil)
	if err != nil {
	        log.Fatal("listenAndServer: ", err)
	}
}

func do(w http.ResponseWriter, r *http.Request) {
	r.ParseForm()
	m, a := r.PostFormValue("m"), r.PostFormValue("a")
	queryForm, err := url.ParseQuery(r.URL.RawQuery)
	if err == nil && len(queryForm["m"]) > 0 {
		m = queryForm["m"][0]
	}
	if err == nil && len(queryForm["a"]) > 0 {
		a = queryForm["a"][0]
	}

	//创建实例
	conf := action.Conf{}
	conf.Op = &m

	//值信息
	v := reflect.ValueOf(conf)
	callMethod(&v, m, []interface{}{a})
}

func callMethod(v *reflect.Value, method string, params []interface{}) {
	//字符串方法调用,且能找到实例属性.Op
	f := (*v).MethodByName(method)
	if f.IsValid() {
		args := make([]reflect.Value, len(params))
		for k, param := range params {
			args[k] = reflect.ValueOf(param)
		}
		//调用
		ret := f.Call(args)
		if ret[0].Kind() == reflect.String {
			fmt.Printf("%s Call result: %s\n", method, ret[0].String())
		}
	} else {
		fmt.Println("can't call " + method)
	}
}


2.在main.go的同一级目录创建action文件夹,里面加入action.go文件
package action

type Conf struct {
	Op       *string `json:"jsonop" xml:"xmlOpName"`
}

func (this Conf) Test(name string) string {
	return *this.Op + name
}

3.编译运行,然后浏览器输入 http://localhost:8080/index?m=Test&a=b
查看输出结果