为了账号安全,请及时绑定邮箱和手机立即绑定

如何过滤 GAE 查询?

如何过滤 GAE 查询?

Go
慕哥6287543 2021-09-27 10:38:09
我正在尝试保存两条记录,然后获取第二条记录。问题是过滤器似乎不起作用。虽然我按姓名(“Andrew W”)过滤,但我总是得到“Joe Citizen”。计数器还指示 2 条记录,而它应该只有 1 条记录。这让我发疯。请参阅下面的完整代码。结果打印counter 2 e2 {"Joe Citizen" "Manager" "2015-03-24 09:08:58.363929 +0000 UTC" ""}package mainimport (    "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)}
查看完整描述

1 回答

?
aluckdog

TA贡献1847条经验 获得超7个赞

(第一)问题是这样的:


q :=  datastore.NewQuery("employee")

q.Filter("Name =", "Andrew W")

Query.Filter()返回包含您指定的过滤器的派生查询。您必须存储返回值并持续使用它:


q := datastore.NewQuery("employee")

q = q.Filter("Name =", "Andrew W")

或者只有一行:


q := datastore.NewQuery("employee").Filter("Name =", "Andrew W")

注意:如果没有这个,您执行的查询将没有过滤器,因此将返回所有以前保存的实体"employee",其中"Joe Citizen"可能是您看到打印的第一个实体。


对于第一次运行,您很可能会看到 0 个结果。请注意,由于您不使用祖先查询,因此最终一致性适用。开发SDK以最终一致性模拟高复制数据存储,因此Put()操作后的查询将看不到结果。


如果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})

但是当然这仅用于测试目的,您不能在生产中执行此操作。


如果想要强一致的结果,请在创建键时指定一个祖先键,并使用祖先查询。仅当您想要高度一致的结果时才需要祖先键。如果您对显示结果的延迟几秒钟没有问题,则不必这样做。另请注意,祖先键不必是现有实体的键,它只是语义。您可以创建任何虚构的密钥。对多个实体使用相同的(虚构的)键会将它们放入同一个实体组中,并且该组上的祖先查询将高度一致。


通常,祖先密钥是现有密钥,通常从当前用户或帐户派生,因为它可以轻松创建/计算,并且它保存/存储一些附加信息,但如上所述,它不一定是。


查看完整回答
反对 回复 2021-09-27
  • 1 回答
  • 0 关注
  • 202 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信