登录
首页 >  Golang >  Go问答

如何过滤 GAE 查询?

来源:Golang技术栈

时间:2023-04-28 21:49:28 227浏览 收藏

今日不肯埋头,明日何以抬头!每日一句努力自己的话哈哈~哈喽,今天我将给大家带来一篇《如何过滤 GAE 查询?》,主要内容是讲解golang等等,感兴趣的朋友可以收藏或者有更好的建议在评论提出,我都会认真看的!大家一起进步,一起学习!

问题内容

我正在尝试保存两条记录,然后获取第二条记录。问题是过滤器似乎不起作用。虽然我按名称(“Andrew W”)过滤,但我总是得到“Joe Citizen”。当它应该只有一个时,计数器还会指示 2 个记录。这让我发疯。请参阅下面的完整代码。结果打印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()返回包含您指定的过滤器的派生查询。您必须存储返回值并持续使用它:

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

或者只有一行:

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

注意:如果没有这个,您执行的查询将没有过滤器,因此将返回所有以前保存的 kind 实体"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})

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

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

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

今天带大家了解了golang的相关知识,希望对你有所帮助;关于Golang的技术知识我们会一点点深入介绍,欢迎大家关注golang学习网公众号,一起学习编程~

声明:本文转载于:Golang技术栈 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>