总所周知,golang现在不支持泛型编程,对于习惯了C++泛型编程思想的程序员来说无疑是一个硬伤。
虽然golang interface{}多少可以满足一些泛型需求,但是无论在效率和编程手法上,都无法达到C++
template那样的灵活性。出于对编译器效率的考虑,官方暂时还没有golang支持泛型的方案。
根据C++泛型的基本思路,我实现了一个名为gpg[1]的工具,使用工具通过文字替换的方式生成需要的.go代码,解决维护多个相似代码的尴尬,通过配置.gpg(ini)实例化模板文件(.gp),替换相关模板参数
达到维护一份代码(.gp),实现类型无关的编程思想。
如源码中example所示,假如需要实现一个未知类型的全局变量,需要在get和set的时候自动加锁,解决不同goroutine访问全局变量时的冲突。按照现有go规则,使用interface{}每次进行类型断言是一个办法,但是应用起来代码不好看,执行效率也没有原生类型快。
还有一个办法是用到什么类型,就把已经实现的代码拷贝一份改一改类型,这样一旦算法有修改,需要修改所有的实现代码,很不靠谱。
gpg可以解决以上所有问题. gpg的使用方法如下:
gpg tool 使用方法:
gpg
-h
显示帮助
gpg
遍历当前目录所有.gpg文件 用同名的.gp文件作为模板生成.go代码文件
gpg 遍历path目录所有.gpg文件 用同名的.gp文件作为模板生成.go代码文件
//////////////////////////////////////////example.gp
//This is an example of using gpg tool for
generic-programming
//this is an example of using gpg to define an auto-lock global value with generic type
//it will be realized to real go code by gpg tool through the .gpg file with the same name
package example
//this is an example of using gpg to define an auto-lock global value with generic type
//it will be realized to real go code by gpg tool through the .gpg file with the same name
package example
import (
"sync"
)
)
//auto locked global value
type AutoLockGbl struct {
val
lock sync.RWMutex
}
type AutoLockGbl struct {
}
//new and init a global value
func New(val ) *AutoLockGbl{
p := &AutoLockGbl{}
p.val = val
return p
}
func New(val ) *AutoLockGbl{
}
//get value, if modify is disable, lock is unneeded
func (me *AutoLockGbl) Get() (r ) {
me.lock.RLock()
defer me.lock.RUnlock()
r = me.val
return
}
func (me *AutoLockGbl) Get() (r ) {
}
//set value, if modify is disable, delete this function
func (me *AutoLockGbl) Set(val ) (r ) {
me.lock.Lock()
defer me.lock.Unlock()
r = me.val
me.val = val
return
}
func (me *AutoLockGbl) Set(val ) (r ) {
}
//////////////////////////////////////////example.gpg
//分别用int,string,uint64实例化example.gp
;this is exactlly an ini file
;it is used to generate code from .gp file
[int]
TYPE_NAME=Int
VALUE_TYPE=int
LOCK_COMMENT=
;it is used to generate code from .gp file
[int]
TYPE_NAME=Int
VALUE_TYPE=int
LOCK_COMMENT=
;禁止修改,不需要锁
[const_str]
TYPE_NAME=ConstStr
VALUE_TYPE=string
LOCK_COMMENT=//
TYPE_NAME=ConstStr
VALUE_TYPE=string
LOCK_COMMENT=//
;禁止修改,不需要锁
[const_u64]
TYPE_NAME=ConstU64
VALUE_TYPE=uint64
LOCK_COMMENT=//
TYPE_NAME=ConstU64
VALUE_TYPE=uint64
LOCK_COMMENT=//
可以得到形如以下的代码文件
/////////////////////////////////example_gpg_int.go
// This file was auto-generated by [gpg] tool
// Last modify at: 2014-01-21 13:06:46.6378366 +0800 +0800
// !!!!!!!!!NEVER MODIFY IT MANUALLY!!!!!!!!!
// Last modify at: 2014-01-21 13:06:46.6378366 +0800 +0800
// !!!!!!!!!NEVER MODIFY IT MANUALLY!!!!!!!!!
//Copyright @Ally 2014. All rights reserved.
//Version: 1.0.0
//Author:
vipally@gmail.com
//Blog site: http://blog.sina.com.cn/ally2014
//Version:
//Author:
//Blog site: http://blog.sina.com.cn/ally2014
//This is an example of using gpg tool for
generic-programming
//this is an example of using gpg to define an auto-lock global value with generic type
//it will be realized to real go code by gpg tool through the .gpg file with the same name
package example
//this is an example of using gpg to define an auto-lock global value with generic type
//it will be realized to real go code by gpg tool through the .gpg file with the same name
package example
import (
"sync"
)
)
//auto locked global value
type AutoLockGblInt struct {
val int
lock sync.RWMutex
}
type AutoLockGblInt struct {
}
//new and init a global value
func NewInt(val int) *AutoLockGblInt{
p := &AutoLockGblInt{}
p.val = val
return p
}
func NewInt(val int) *AutoLockGblInt{
}
//get value, if modify is disable, lock is unneeded
func (me *AutoLockGblInt) Get() (r int) {
me.lock.RLock()
defer me.lock.RUnlock()
r = me.val
return
}
func (me *AutoLockGblInt) Get() (r int) {
}
//set value, if modify is disable, delete this function
func (me *AutoLockGblInt) Set(val int) (r int) {
me.lock.Lock()
defer me.lock.Unlock()
r = me.val
me.val = val
return
}
func (me *AutoLockGblInt) Set(val int) (r int) {
}
/////////////////////////////////example_gpg_const_str.go
// This file was auto-generated by [gpg] tool
// Last modify at: 2014-01-21 13:06:46.6408372 +0800 +0800
// !!!!!!!!!NEVER MODIFY IT MANUALLY!!!!!!!!!
// Last modify at: 2014-01-21 13:06:46.6408372 +0800 +0800
// !!!!!!!!!NEVER MODIFY IT MANUALLY!!!!!!!!!
//Copyright @Ally 2014. All rights reserved.
//Version: 1.0.0
//Author:
vipally@gmail.com
//Blog site: http://blog.sina.com.cn/ally2014
//Version:
//Author:
//Blog site: http://blog.sina.com.cn/ally2014
//This is an example of using gpg tool for
generic-programming
//this is an example of using gpg to define an auto-lock global value with generic type
//it will be realized to real go code by gpg tool through the .gpg file with the same name
package example
//this is an example of using gpg to define an auto-lock global value with generic type
//it will be realized to real go code by gpg tool through the .gpg file with the same name
package example
import (
//"sync"
)
)
//auto locked global value
type AutoLockGblConstStr struct {
val string
//lock sync.RWMutex
}
type AutoLockGblConstStr struct {
}
//new and init a global value
func NewConstStr(val string) *AutoLockGblConstStr{
p := &AutoLockGblConstStr{}
p.val = val
return p
}
func NewConstStr(val string) *AutoLockGblConstStr{
}
//get value, if modify is disable, lock is unneeded
func (me *AutoLockGblConstStr) Get() (r string) {
// me.lock.RLock()
// defer me.lock.RUnlock()
r = me.val
return
}
func (me *AutoLockGblConstStr) Get() (r string) {
//
//
}
//set value, if modify is disable, delete this function
//func (me *AutoLockGblConstStr) Set(val string) (r string) {
// me.lock.Lock()
// defer me.lock.Unlock()
// r = me.val
// me.val = val
// return
//}
//func (me *AutoLockGblConstStr) Set(val string) (r string) {
//
//
//
//
//
//}
这样基本可以满足golang对泛型的需求,而不需要付出额外代价。
有疑问加站长微信联系(非本文作者)