我正在尝试保存两条记录,然后获得第二条记录。问题在于过滤器似乎无法正常工作。尽管我按名称(“ Andrew W”)进行过滤,但我总是得到“ Joe Citizen”。该计数器还指示2条记录,当它应仅为1条时。这让我发疯。请参阅下面的完整代码。结果打印counter 2 e2 {"Joe Citizen" "Manager" "2015-03-24 09:08:58.363929 +0000 UTC" ""}
counter 2 e2 {"Joe Citizen" "Manager" "2015-03-24 09:08:58.363929 +0000 UTC" ""}
package main import ( "fmt" "time" "net/http" "google.golang.org/appengine" "google.golang.org/appengine/datastore" ) type Employee struct { Name string Role string HireDate time.Time Account string } func init(){ http.HandleFunc("/", handle) } func handle(w http.ResponseWriter, r *http.Request) { c := appengine.NewContext(r) e1 := Employee{ Name: "Joe Citizen", Role: "Manager", HireDate: time.Now(), } _, err := datastore.Put(c, datastore.NewKey(c, "employee", "", 0, nil), &e1) if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) panic(err) return } e1.Name = "Andrew W" _, err = datastore.Put(c, datastore.NewKey(c, "employee", "", 0, nil), &e1) if err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) panic(err) return } var e2 Employee q := datastore.NewQuery("employee") q.Filter("Name =", "Andrew W") cnt, err := q.Count(c) if err !=nil{ http.Error(w, err.Error(), http.StatusInternalServerError) panic(err) return } for t := q.Run(c); ; { if _, err := t.Next(&e2); err != nil { http.Error(w, err.Error(), http.StatusInternalServerError) panic(err) return } break } fmt.Fprintf(w, "counter %v e2 %q", cnt, e2) }
(第一个)问题是:
q := datastore.NewQuery("employee") q.Filter("Name =", "Andrew W")
Query.Filter()返回包含您指定的过滤器的派生查询。您必须存储返回值并持续使用它:
Query.Filter()
q := datastore.NewQuery("employee") q = q.Filter("Name =", "Andrew W")
或仅一行:
q := datastore.NewQuery("employee").Filter("Name =", "Andrew W")
注意:没有此查询,您执行的查询将没有过滤器,因此将返回所有先前保存的那种实体"employee",其中"Joe Citizen"可能是您看到的第一个打印的实体。
"employee"
"Joe Citizen"
对于第一次运行,您很可能会看到0个结果。请注意,由于您不使用Ancestor查询,因此 最终的一致性适用 。开发SDK会模拟高复制数据存储库及其最终一致性,因此Put()操作后的查询将看不到结果。
Put()
如果time.Sleep()在进行查询之前先输入少量内容,则会看到预期的结果:
time.Sleep()
time.Sleep(time.Second) var e2 Employee q := datastore.NewQuery("employee").Filter("Name=", "Andrew W") // Rest of your code...
还要注意,在SDK中运行代码可以通过创建如下上下文来模拟 强一致性 :
c, err := aetest.NewContext(&aetest.Options{StronglyConsistentDatastore: true})
但这当然只是出于测试目的,您不能在生产中这样做。
如果要获得高度一致的结果,请在创建键时指定一个祖先键,然后使用祖先查询。仅当您想要非常一致的结果时才需要祖先键。如果您满意,但可以延迟几秒钟才能显示结果,则不必这样做。还要注意,祖先键不必是现有实体的键,而只是语义。您可以创建任何虚构的密钥。对多个实体使用相同的(虚构的)键会将它们放入同一实体组,并且该组上的祖先查询将保持高度一致。
祖先键通常是一个现有键,通常是从当前用户或帐户派生的,因为它可以轻松创建/计算,并且可以保存/存储一些其他信息,但是如上所述,不必如此。