一尘不染

Go SQL-预备语句范围

go

我正在使用Go(1.6.x)sql包以及PostGres(9.4)构建API。我准备好的陈述应具有适用范围或要求范围吗?阅读文档后,将它们放在应用程序范围内似乎可以更有效地减少准备阶段的数量。但是,也许还有其他考虑因素,而准备好的陈述并不是为了活那么久?


阅读 228

收藏
2020-07-02

共1个答案

一尘不染

准备好的语句使您可以执行重复的SQL命令,这些命令可能仅在参数值方面有所不同。

它们并不意味着“长寿”,因为一条准备好的语句可能会保留一个活动的数据库连接(如果从事务中调用,它们会保留)(“长寿”意味着不使用它们时;重复执行许多条准备好的语句是完全可以的)时间,即使这将需要很长时间)。连接是一种昂贵的资源,应仅在需要时才保留。仅通过创建一堆准备好的语句而不关闭它们,您可能会耗尽活动/允许的连接,然后阻止与数据库服务器的进一步通信。

如果要在同一(HTTP)请求中多次执行相同的insertupdateselect使用不同参数的语句,请使用预处理语句。不要使用准备好的语句来延长(HTTP)请求。

在某些驱动程序实现和数据库服务器中,准备好的语句还可能涉及在DB服务器本身(而不是Go应用程序)中分配的资源。例如,一条准备好的语句可以在DB服务器上进行预编译,并且服务器可以准备查询执行计划,为其分配某些资源(例如内存)。这些可能会永久保留,直到准备好的语句关闭。

有一篇文章(由Myles McDonnell在下面的评论中发布),涉及Go 中Prepared Statements的实现细节。它提到如果未从事务创建准备好的语句,它们会将连接释放回连接池,但是在需要时,它们会尝试重用准备好的语句(因为如果db服务器辅助/也起着积极的作用,在准备好的语句中,它绑定到服务器端的连接)。如果没有,他们将在新的连接上
重新准备 它们(导致不希望的性能开销)。

总而言之,您所描述的是一个工作模型,如果在许多后续请求中需要/执行的准备好的语句数量很少,那么它们可能意味着响应时间更短。但这也意味着从长远来看,它们可能会导致所有准备好的语句将在池的所有连接上准备好。确定您的情况是否可以接受。

通常,应该避免这种情况(并且在HTTP请求结束之前关闭预处理语句),但是如果您只有少数几个,并且确实需要多个彼此紧接的请求,则可以将它们移出请求范围。

2020-07-02