我想在 BYTEA 列中插入一些二进制数据
我将如何将 somefile.tar.gz 的内容插入到具有 BYTEA 列的表中?
- 一个例子会有所帮助
- 是否可以使用 BYTEA 的 HEX 格式(postgres 9.2)?
- 是否可以从/到 golang 向/从 postgres 流式传输?
我想在 BYTEA 列中插入一些二进制数据
我将如何将 somefile.tar.gz 的内容插入到具有 BYTEA 列的表中?
使用 sqlx 和 lib/pq 你只需使用一个字节片[]byte
:
// data is []byte
data, err := os.ReadFile("somefile.tar.gz")
if err != nil {
panic(err)
}
_, err = db.Exec("INSERT INTO test (id, data) values ($1, $2)", 1, data)
if err != nil {
panic(err)
}
注意:如果您使用字符串文字对此进行测试,请记住 Go 字符串(除非另有说明)以 UTF-8 编码,因此您将存储在BYTEA
列中的字节将是该字符串逐字的 UTF-8 编码. 例如[]byte("你好")
商店e4 bd a0 e5 a5 bd
。
是否可以使用 BYTEA 的 HEX 格式(postgres 9.2)?
是的,您可以在 SQL 字符串中使用十六进制格式。除非字符串包含在反引号中`
,否则必须转义斜杠。
_, err = db.Exec("INSERT INTO test (id, data) values ($1, '\\xBAADC0DE')", 2)
if err != nil {
panic(err)
}
与将 Go 字符串转换为字节切片不同,使用 Postgres 十六进制格式存储文字字节,例如ba ad c0 de
. 这不会是 UTF-8 编码的。
是否可以从/到 golang 向/从 postgres 流式传输?
如果愿意切换到github.com/jackc/pgx
,可以流式传输大对象(PostgreSQL文档)。该pgx.LargeObject
类型实现:
io.Writer io.Reader io.Seeker io.Closer
大对象存储在系统表中,没有可以在表的列中使用的大对象类型。大对象由它们的对象标识符引用。所以需要用文件元数据和oid
映射来维护一个单独的表。
示例程序:
package main
import (
"context"
"io"
"log"
"os"
"time"
"github.com/jackc/pgx/v4"
)
const (
// files table maps Large Object oid to file names
createFileTable = `CREATE TABLE files (
id oid primary key,
name varchar,
unique(name)
);`
)
func main() {
ctx, cancel := context.WithTimeout(context.TODO(), time.Minute)
defer cancel()
conn, err := pgx.Connect(ctx, "user=postgres host=/run/postgresql dbname=postgres")
if err != nil {
panic(err)
}
defer conn.Close(ctx)
if _, err = conn.Exec(ctx, createFileTable); err != nil {
panic(err)
}
written, err := storeFile(ctx, conn, "somefile.bin")
log.Printf("storeFile written: %d", written)
if err != nil {
panic(err)
}
read, err := loadFile(ctx, conn, "somefile.bin")
log.Printf("loadFile read: %d", read)
if err != nil {
panic(err)
}
}
// storeFile as Large Object in the database.
// The resulting object identifier is stored along with the file name in the files table.
// The amount of written bytes and an erorr is returned, if one occured.
func storeFile(ctx context.Context, conn *pgx.Conn, name string) (written int64, err error) {
file, err := os.Open(name)
if err != nil {
return 0, err
}
defer file.Close()
// LargeObjects can only operate on an active TX
tx, err := conn.Begin(ctx)
if err != nil {
return 0, err
}
defer tx.Rollback(ctx)
lobs := tx.LargeObjects()
// Create a new Large Object.
// We pass 0, so the DB can pick an available oid for us.
oid, err := lobs.Create(ctx, 0)
if err != nil {
return 0, err
}
// record the oid and filename in the files table
_, err = tx.Exec(ctx, "INSERT INTO files (id, name) VALUES ($1, $2)", oid, name)
if err != nil {
return 0, err
}
// Open the new Object for writing.
obj, err := lobs.Open(ctx, oid, pgx.LargeObjectModeWrite)
if err != nil {
return 0, err
}
// Copy the file stream to the Large Object stream
written, err = io.Copy(obj, file)
if err != nil {
return written, err
}
err = tx.Commit(ctx)
return written, err
}
// loadFile loads the file identified by name as Large Object
// and writes the contents to a local file by the same name.
// The amount of bytes read or an error is returned.
func loadFile(ctx context.Context, conn *pgx.Conn, name string) (read int64, err error) {
tx, err := conn.Begin(ctx)
if err != nil {
return 0, err
}
defer tx.Rollback(ctx)
var oid uint32
err = conn.QueryRow(ctx, "SELECT id FROM files WHERE name = $1", name).Scan(&oid)
if err != nil {
return 0, err
}
file, err := os.Create(name)
if err != nil {
return 0, err
}
lobs := tx.LargeObjects()
obj, err := lobs.Open(ctx, oid, pgx.LargeObjectModeRead)
if err != nil {
return 0, err
}
return io.Copy(file, obj)
}