我想做一些像python在app引擎上支持的Expando Model。
有时您不想提前申报您的财产。一个 特殊模型子类Expando改变其实体的行为 以便分配任何属性(只要它不以一个开头 下划线)保存到数据存储区。
我怎样才能在Go中执行此操作?
答案 0 :(得分:6)
事先注意:
有2个API。具有导入路径appengine/datastore
的那个使用通道作为参数。导入路径google.golang.org/appengine/datastore
的另一个使用切片。根据您的情况调整以下示例。有关详细信息,请参阅此问题:How to correctly import Golang appengine?
具有动态属性的实体的关键是PropertyLoadSaver
接口。通过实现此接口,您可以动态 - 在节省时间 - 构建要保存的实体的属性。
此外,为了不必自己执行此操作,Go AppEngine平台提供了PropertyList
类型,它基本上是属性Property
的列表(切片),它还实现了PropertyLoadSaver
。< / p>
所以Go中的Expando模型是PropertyList
。只需添加您希望实体拥有的属性,并保存此PropertyList
值。
以下是一个例子:
c := appengine.NewContext(r)
props := datastore.PropertyList{
datastore.Property{Name: "time", Value: time.Now()},
datastore.Property{Name: "email", Value: "me@myhost.com"},
}
k := datastore.NewIncompleteKey(c, "DynEntity", nil)
key, err := datastore.Put(c, k, &props)
c.Infof("%v %v", key, err)
此示例使用2个动态属性保存名为"DynEntity"
的实体:"time"
和"email"
。
由于PropertyList
类型是切片,您还可以使用内置append()
函数向其添加属性,因此您也可以像这样初始化props
:
var props datastore.PropertyList
props = append(props, datastore.Property{Name:"time", Value: time.Now()})
props = append(props, datastore.Property{Name:"email", Value: "me@myhost.com"})
map
转变为动态实体 PropertyLoadSaver
界面并不复杂,我们可以自己实现。在以下示例中,我在自定义类型上实现它,这是一个简单的map
:
type DynEnt map[string]interface{}
func (d *DynEnt) Load(props []datastore.Property) error {
// Note: you might want to clear current values from the map or create a new map
for _, p := range props {
(*d)[p.Name] = p.Value
}
return nil
}
func (d *DynEnt) Save() (props []datastore.Property, err error) {
for k, v := range *d {
props = append(props, datastore.Property{Name: k, Value: v})
}
return
}
以下是使用通道而不是切片的“旧”接口的实现方式:
type DynEnt map[string]interface{}
func (d *DynEnt) Load(ch <-chan datastore.Property) error {
// Note: you might want to clear current values from the map or create a new map
for p := range ch { // Read until channel is closed
(*d)[p.Name] = p.Value
}
return nil
}
func (d *DynEnt) Save(ch chan<- datastore.Property) error {
defer close(ch) // Channel must be closed
for k, v := range *d {
ch <- datastore.Property{Name: k, Value: v}
}
return nil
}
现在我们可以像使用Go中的任何其他地图一样使用我们的DynEnt
类型,并且由于它实现了PropertyLoadSaver
,因此可以将其保存为实体(以及任何实体可加载到其中):
c := appengine.NewContext(r)
d := DynEnt{"email": "me@myhost.com", "time": time.Now()}
k := datastore.NewIncompleteKey(c, "DynEntity", nil)
key, err := datastore.Put(c, k, &d)
c.Infof("%v %v", key, err)