本库通过定义json文件,使用工具生成redis orm类文件
可以处理1对1类型的数据、以及1对N类型的数据
已本项目例子为例:
redis2go --input_dir=./example/redis_def --output_dir=./example --package=main
redis2go 二进制程序不再提供,请使用 docker 镜像,比如:
docker run --rm -v $PWD/redis_def:/app/input -v $PWD:/app/output fananchong/redis2go --package=main
- 支持 golang 基本类型
- 支持结构体,具体如下:
编码 | json 文件定义例子 |
---|---|
encoding/json | example/redis_def/TestStruct1.json 、 example/redis_def/TestStruct2.json |
protobuf | example/redis_def/TestStruct7.json 、 example/redis_def/TestStruct8.json |
cstruct-go | example/redis_def/TestStruct5.json 、 example/redis_def/TestStruct6.json |
gogoprotobuf | 类型使用 gogo ,这里不再举例 |
例子可以参考:example
即1个ID对应一条数据。
工具生成的redis操作类,有以下API:
API | 说明 |
---|---|
NewXXX | 构建一个XXX redis操作类 |
Load | 从数据库加载数据 |
Save | 保存被改变的数据 |
Delete | 删除本条数据 |
SetYYY | 修改YYY属性 |
GetYYY | 获取YYY属性 |
HasKey | 是否有某条数据 |
DirtyData | 获取脏数据 |
Save2 | 以脏数据为参数,保存操作 |
1对1类型,redis操作类,映射到redis hashmap上。
每次保存,只更新被修改的字段。因此效率是很好的。
func test11() {
dbName := "db1"
go_redis_orm.SetNewRedisHandler(go_redis_orm.NewDefaultRedisClient)
go_redis_orm.CreateDB(dbName, []string{"192.168.1.12:16379"}, "", 0)
// key值为1的 TestStruct1 数据
data1 := NewTestStruct1(dbName, 1)
data1.SetMyb(true)
data1.SetMyf1(1.5)
data1.SetMyi5(100)
data1.SetMys1("hello")
data1.SetMys2([]byte("world"))
err := data1.Save()
if err != nil {
panic(err)
}
data2 := NewTestStruct1(dbName, 1)
err = data2.Load()
if err == nil {
if data2.GetMyb() != true ||
data2.GetMyf1() != 1.5 ||
data2.GetMyi5() != 100 ||
data2.GetMys1() != "hello" ||
string(data2.GetMys2()) != "world" {
panic("#1")
}
} else {
panic(err)
}
err = data2.Delete()
if err != nil {
panic(err)
}
var hasKey int
hasKey, err = data2.HasKey()
if hasKey != 0 {
panic("#2")
}
fmt.Println("OK")
}
即1个ID对应N条数据。
工具生成的redis操作类,除了上面的API,还有以下API:
API | 说明 |
---|---|
NewItem | 增加1条数据 |
GetItem | 获取某条数据 |
DeleteItem | 删除某条数据 |
GetItems | 获取所有条数据 |
DirtyData | 获取脏数据 |
Save2 | 以脏数据为参数,保存操作 |
1对N类型,redis操作类,映射到redis hashmap上。
在hashmap上是这样的: key : itemId1-itemData1, itemId2-itemData2 ... itemIdN-itemDataN
考虑过把N条数据打包一起存储(即1对1的方式),也考虑过把每条数据全部打碎,变成hashmap的field。
这2种方法均存在性能上的问题。因此采用折中的方案:数据ID作为field,数据内容打包作为field的值。
这样每次保存,只更新被修改的那条数据。同时field数量相对的少1个数量级。
友情提示:如果业务需求,N值很大,这样的数据是不适合使用redis存储的。最好把N限制在小于一两百条数据的应用场景。
func test1n() {
dbName := "db2"
go_redis_orm.SetNewRedisHandler(go_redis_orm.NewDefaultRedisClient)
go_redis_orm.CreateDB(dbName, []string{"192.168.1.12:16379"}, "", 0)
data1 := NewTestStruct2(dbName, 8)
item1 := data1.NewItem(1)
item1.SetMyf2(99.9)
item2 := data1.NewItem(2)
item2.SetMys1("hello")
item2.SetMys2([]byte("world"))
err := data1.Save()
if err != nil {
panic(err)
}
data2 := NewTestStruct2(dbName, 8)
err = data2.Load()
if err != nil {
panic(err)
}
fmt.Printf("2: %+v\n", data2.GetItem(1))
fmt.Printf("2: %+v\n", data2.GetItem(2))
data2.DeleteItem(1)
data2.Save()
data3 := NewTestStruct2(dbName, 8)
data3.Load()
for _, v := range data3.GetItems() {
fmt.Printf("3: %+v\n", v)
}
data3.Delete()
data3.Save()
data4 := NewTestStruct2(dbName, 8)
data4.Load()
fmt.Printf("4: item count = %d\n", len(data4.GetItems()))
fmt.Println("OK")
}
-
定义 json 文件
格式参考:example/redis_def
-
生成 go 文件,参考 example/g.sh
执行下列语句:
./build.sh
本库支持第3方redis客户端整合进本项目,通过调用go_redis_orm.SetNewRedisHandler函数
需要实现go_redis_orm.IClient接口
type IClient interface {
Do(commandName string, args ...interface{}) (reply interface{}, err error)
}
例子参考:
详细参见:http://blog.csdn.net/u013272009/article/details/78513251
部分程序,可能有多协程读写的需求。
可以使用如下接口:
- DirtyData()
- Save2()
v1.1 版本将实现以下内容:
- 高并发访问 redis
- 保证同键值数据访问 redis 无时序问题
- 性能测试,并附性能结果
- 提供按实例化 go-redis-orm 对象,而非单单一个全局实例