首页 > 解决方案 > 如果关闭了 Go sql.DB,是否会关闭任何未关闭的准备好的查询?

问题描述

在使用的 Go 程序中,当我使用 Postgres DBdatabase/sql时,是否会关闭任何未关闭的准备好的查询?Close

我已将其简化为一个非常简单的示例,该示例不需要Prepare但仍显示问题(我相信我可以将查询字符串传递给QueryRow并获得隐含的Prepare,但在此处将其显式保留,以便我可以提出问题):

import (
    "database/sql"
)

// Store struct is the postgres
type Store struct {
    pq *sql.DB
}

type Info struct {
    ID      string `json:"id"`
    Name    string `json:"name"`
}

func (s *Store) GetInfo(id string) Info {
    defer s.pq.Close()
    stmt, err := s.pq.Prepare(`
            SELECT id, name 
            FROM info 
            WHERE id = $1 LIMIT 1
            `)
    if err != nil {
        return Info{}
    }
    var res Info
    _ = stmt.QueryRow(id).Scan(&res.ID, &res.Name)
    return res
}

标签: postgresqlgoresource-leak

解决方案


从技术上讲database/sql ,绝对希望您关闭自己准备好的语句,并且在关闭 DB 或 DC 时不会为您执行此操作。此外,我认为当您的程序退出时服务器可能会清理后端内存,但 PostgreSQL 也不会清理它......

https://github.com/lib/pq/issues/446

如果你得到隐式 Prepare thendatabase/sql将为你处理清理,但如果你一遍又一遍地运行这些查询,效率会降低,所以我强烈建议你自己清理:

defer stmt.Close()

或类似的。


推荐阅读