本博客原创博文版权所有 @Allyvipally@
仅供交流学习使用用于商业用途请联系原作者
转载请注明出处:/ally2014
总所周知,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代码文件
//
//This is an example of using gpg tool forgeneric-programming
//this is an example of using gpg to define an auto-lock globalvalue with generic type
//it will be realized to real go code by gpg tool through the .gpgfile with the same name
package example
//this is an example of using gpg to define an auto-lock globalvalue with generic type
//it will be realized to real go code by gpg tool through the .gpgfile with the same name
package example
import (
"sync"
)
)
//auto locked global value
type AutoLockGbl struct {
val
lock
}
type AutoLockGbl struct {
}
//new and init a global value
func New(val ) *AutoLockGbl{
p := &AutoLockGbl{}
= val
return p
}
func New(val ) *AutoLockGbl{
}
//get value, if modify is disable, lock is unneeded
func (me *AutoLockGbl) Get() (r ) {
()
defer ()
r =
return
}
func (me *AutoLockGbl) Get() (r ) {
}
//set value, if modify is disable, delete this function
func (me *AutoLockGbl) Set(val ) (r ) {
()
defer ()
r =
= val
return
}
func (me *AutoLockGbl) Set(val ) (r ) {
}
//
//分别用int,string,uint64实例化
;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@
//Blog site: /ally2014
//Version:
//Author:
//Blog site: /ally2014
//This is an example of using gpg tool forgeneric-programming
//this is an example of using gpg to define an auto-lock globalvalue with generic type
//it will be realized to real go code by gpg tool through the .gpgfile with the same name
package example
//this is an example of using gpg to define an auto-lock globalvalue with generic type
//it will be realized to real go code by gpg tool through the .gpgfile with the same name
package example
import (
"sync"
)
)
//auto locked global value
type AutoLockGblInt struct {
val
int
lock
}
type AutoLockGblInt struct {
}
//new and init a global value
func NewInt(val int) *AutoLockGblInt{
p := &AutoLockGblInt{}
= val
return p
}
func NewInt(val int) *AutoLockGblInt{
}
//get value, if modify is disable, lock is unneeded
func (me *AutoLockGblInt) Get() (r int) {
()
defer ()
r =
return
}
func (me *AutoLockGblInt) Get() (r int) {
}
//set value, if modify is disable, delete this function
func (me *AutoLockGblInt) Set(val int) (r int) {
()
defer ()
r =
= 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@
//Blog site: /ally2014
//Version:
//Author:
//Blog site: /ally2014
//This is an example of using gpg tool forgeneric-programming
//this is an example of using gpg to define an auto-lock globalvalue with generic type
//it will be realized to real go code by gpg tool through the .gpgfile with the same name
package example
//this is an example of using gpg to define an auto-lock globalvalue with generic type
//it will be realized to real go code by gpg tool through the .gpgfile with the same name
package example
import (
//"sync"
)
)
//auto locked global value
type AutoLockGblConstStr struct {
val
string
//lock
}
type AutoLockGblConstStr struct {
}
//new and init a global value
func NewConstStr(val string) *AutoLockGblConstStr{
p := &AutoLockGblConstStr{}
= val
return p
}
func NewConstStr(val string) *AutoLockGblConstStr{
}
//get value, if modify is disable, lock is unneeded
func (me *AutoLockGblConstStr) Get() (r string) {
// ()
// defer ()
r =
return
}
func (me *AutoLockGblConstStr) Get() (r string) {
//
//
}
//set value, if modify is disable, delete this function
//func (me *AutoLockGblConstStr) Set(val string) (r string) {
// ()
// defer ()
// r =
// = val
// return
//}
//func (me *AutoLockGblConstStr) Set(val string) (r string) {
//
//
//
//
//
//}
这样基本可以满足golang对泛型的需求,而不需要付出额外代价。
本博客原创博文版权所有 @Allyvipally@
仅供交流学习使用用于商业用途请联系原作者
转载请注明出处:/ally2014