使用 Golang 构建可维护的 SQL 查询
任何使用 SQL 查询的应用程序都可以受益于使用查询生成器来提高代码的可读性、可维护性和安全性。事实上,Golang 中有许多不同的库可以做到这一点。在 Vaunt,我们尝试了许多不同的选择,最后决定自己创建一个。最终,我们想要一些安全的东西,并提供变量替换来防止 SQL 注入,同时仍然可读并且能够有条件语句。因此,我们创建了一个名为 tqla 的新库,并于去年年底发布并宣布。您可以在这篇文章中阅读更多相关信息。
在构建 tqla 之前,我们主要使用 Squirrel 来构建 SQL 查询逻辑——我们强烈推荐它。我们仍然在某些领域使用 Squirrel,但已逐渐开始用 tqla 替换和实现新的查询构建逻辑。我们发现在许多实例中,tqla 提高了我们维护代码和修复使用其他语句生成器时遇到的问题的能力。
现实世界用例
在 Vaunt,我们最近进行了从 CockroachDB 到 TiDB 的数据库迁移。虽然 CockroachDB 高性能且可靠,但我们最终决定添加到我们的技术堆栈中以支持 OLAP 数据库。这样做的需要是支持我们对开源社区洞察产品的分析工作量。为了保持较小的技术足迹,我们决定继续使用 TiDB 并利用该数据库的 HTAP 架构。
CockroachDB 与 PostgreSQL 很大程度上兼容,我们的许多 SQL 查询都使用 PostgreSQL 语法。要切换到 TiDB,我们必须更改一些表并更新查询以使用 MySQL 语法。在迁移过程中的一些位置,我们发现我们不正确地使用条件查询构建语句,并且缺乏适当的测试来发现语句生成不正确。
示范
在 Squirrel 的自述文件中,有一个示例说明如何使用条件查询构建来更新具有可选过滤器的语句:
if len(q) > 0 { users = users.Where("name LIKE ?", fmt.Sprint("%", q, "%")) }
这是一个真实但简化的示例,说明我们如何更新其中一个查询以有条件连接表并添加可选过滤器:
psql := squirrel.StatementBuilder.PlaceholderFormat(squirrel.Question) statementBuilder := psql.Select(`i.id`). From("vaunt.installations i"). Where(`entity_name = ?`, name) if len(provider) > 0 { statementBuilder.Where(`provider = ?`, provider) } if len(repo) > 0 { statementBuilder.Join(`repositories as r on JSON_CONTAINS(i.repositories, CONCAT('["', r.id, '"]'))`) statementBuilder.Where(`r.name = ?`, repo) }
你能发现代码的问题吗?如果没有,请不要担心——在我们运行测试之前,我们自己的代码审查也会忽略这一点。
这里的问题是我们忘记使用构建器函数的结果更新语句构建器。例如,提供者条件过滤器应改为:
if len(provider) > 0 { statementBuilder = statementBuilder.Where(`provider = ?`, provider) }
这是一个相对简单的错误,可以通过足够的测试用例轻松发现,但由于它不是技术上无效的代码,因此可能需要一些时间才能立即意识到发生了什么。
此设置的另一个可读性问题是条件连接与初始 select 语句是分开的。我们可以重新组织构建器,将每个部分放在它应该去的地方,但这需要多次重复的条件语句检查,并且仍然会遇到一些可读性问题。
使用tqla
上面使用 Squirrel 的演示已被重写,tqla 中的等效项如下所示:
t, err := tqla.New(tqla.WithPlaceHolder(tqla.Question)) if err != nil { return nil, err } query, args, err := t.Compile(` SELECT i.id FROM vaunt.installations as i {{ if .Repo }} JOIN vaunt.repositories as r on JSON_CONTAINS(i.repositories, CONCAT('["', r.id, '"]'), '$') {{ end }} WHERE entity_name = {{ .Name}} {{ if .Provider }} AND i.provider = {{ .Provider }} {{ end }} {{ if .Repo }} AND r.name = {{ .Repo }} {{ end }} `, data) if err != nil { return nil, err }
如您所见,tqla 的模板语法使得合并条件子句变得非常简单。 Tqla 自动用指定的占位符替换我们设置的变量,并提供我们可以与 sql 驱动程序一起使用来执行语句的参数。
与 Squirrel 类似,这种语句构建方法很容易测试,因为我们可以创建不同的数据对象集来传递给模板构建器并验证输出。
您可以看到,我们可以轻松地将查询的条件部分添加到最适合的位置。例如,这里我们在 FROM 语句之后直接有一个条件 JOIN,尽管我们仍然有多个条件检查,但它并没有使模板过于复杂。
自定义功能
另一个有助于提高 sql 构建器可维护性的不错的 tqla 功能是能够定义我们可以在模板中使用的自定义函数来抽象一些转换逻辑。
下面是我们如何使用函数将 Golang 的 time.Time 值转换为 sql.NullTime 的示例,以便我们无需事先转换即可对数据对象进行插入:
funcs := template.FuncMap{ "time": func(t time.Time) sql.NullTime { if t.IsZero() { return sql.NullTime{Valid: false} } return sql.NullTime{Time: t, Valid: true} }, } t, err := tqla.New(tqla.WithPlaceHolder(tqla.Question), tqla.WithFuncMap(funcs)) if err != nil { return err }
通过在 tqla funcs 映射中定义此函数,我们现在可以通过向其提供来自数据对象(即 time.Time 字段)的参数来在查询模板中自由使用它。我们甚至可以在同一模板中使用不同字段多次调用此函数。
Here is a simplified example:
statement, args, err := t.Compile(` INSERT INTO events (name, created_at, merged_at, closed_at) VALUES ( {{ .Name }}, {{ time .CreatedAt }}, {{ time .MergedAt }}, {{ time .ClosedAt }} )`, eventData)
Conclusion
In conclusion, we believe that using tqla can help improve the maintainability of query building logic while offering some powerful utility for creating dynamic queries. The simplicity of the template structure allows for clean code readability and can make it faster to debug any potential errors.
We made tqla open source to share this library in hopes that it provides a good option for other users wanting a simple, maintainable, and secure way to build sql queries in many different types of applications.
If you are interested, please check out the repository and give it a star if it helps you in any way. Feel free to make any feature requests or bug reports!
We are always open to receiving feedback and contributions.
To stay in the loop on future development, follow us on X or join our Discord!
以上是使用 Golang 构建可维护的 SQL 查询的详细内容。更多信息请关注PHP中文网其他相关文章!

热AI工具

Undress AI Tool
免费脱衣服图片

Undresser.AI Undress
人工智能驱动的应用程序,用于创建逼真的裸体照片

AI Clothes Remover
用于从照片中去除衣服的在线人工智能工具。

Clothoff.io
AI脱衣机

Video Face Swap
使用我们完全免费的人工智能换脸工具轻松在任何视频中换脸!

热门文章

热工具

记事本++7.3.1
好用且免费的代码编辑器

SublimeText3汉化版
中文版,非常好用

禅工作室 13.0.1
功能强大的PHP集成开发环境

Dreamweaver CS6
视觉化网页开发工具

SublimeText3 Mac版
神级代码编辑软件(SublimeText3)

搭建一个用Go编写的Web服务器并不难,核心在于利用net/http包实现基础服务。1.使用net/http启动最简服务器:通过几行代码注册处理函数并监听端口;2.路由管理:使用ServeMux组织多个接口路径,便于结构化管理;3.常见做法:按功能模块分组路由,并可用第三方库支持复杂匹配;4.静态文件服务:通过http.FileServer提供HTML、CSS和JS文件;5.性能与安全:启用HTTPS、限制请求体大小、设置超时时间以提升安全性与性能。掌握这些要点后,扩展功能将更加容易。

音视频处理的核心在于理解基本流程与优化方法。1.其基本流程包括采集、编码、传输、解码和播放,每个环节均有技术难点;2.常见问题如音画不同步、卡顿延迟、声音噪音、画面模糊等,可通过同步调整、编码优化、降噪模块、参数调节等方式解决;3.推荐使用FFmpeg、OpenCV、WebRTC、GStreamer等工具实现功能;4.性能管理方面应注重硬件加速、合理设置分辨率帧率、控制并发及内存泄漏问题。掌握这些关键点有助于提升开发效率和用户体验。

编写KubernetesOperator的最有效方式是使用Go语言结合Kubebuilder和controller-runtime。1.理解Operator模式:通过CRD定义自定义资源,编写控制器监听资源变化并执行调和循环以维护期望状态。2.使用Kubebuilder初始化项目并创建API,自动生成CRD、控制器和配置文件。3.在api/v1/myapp_types.go中定义CRD的Spec和Status结构体,运行makemanifests生成CRDYAML。4.在控制器的Reconcil

TooptimizeGoapplicationsinteractingwithPostgreSQLorMySQL,focusonindexing,selectivequeries,connectionhandling,caching,andORMefficiency.1)Useproperindexing—identifyfrequentlyqueriedcolumns,addindexesselectively,andusecompositeindexesformulti-columnquer

Go的通道分为缓冲和非缓冲两种,非缓冲通道要求发送和接收同时准备好,适合协调同步操作;而缓冲通道允许数据暂存,适合解耦系统组件。使用时需注意避免向满缓冲通道写入或从空通道读取导致阻塞,以及适时关闭通道以通知接收方数据结束。

使用fmt.Scanf可读取格式化输入,适合简单结构化数据,但字符串遇空格截止;2.推荐使用bufio.Scanner逐行读取,支持多行输入、EOF检测和管道输入,并可处理扫描错误;3.使用io.ReadAll(os.Stdin)一次性读取全部输入,适用于处理大块数据或文件流;4.实时按键响应需第三方库如golang.org/x/term,常规场景使用bufio已足够;实际建议:交互式简单输入用fmt.Scan,行输入或管道用bufio.Scanner,大块数据用io.ReadAll,且始终处理

OAuth2实现分为客户端和服务端。客户端使用golang.org/x/oauth2包,步骤为:1.引入包;2.配置客户端信息并构建Config对象;3.生成授权链接;4.处理回调获取Token;5.构造带授权的HTTP客户端。服务端以go-oauth2/oauth2为例,流程包括:1.初始化存储;2.设置客户端信息;3.创建OAuth2服务实例;4.编写路由处理授权和Token请求。注意事项有:跨域问题、状态校验、启用HTTPS、Token有效期管理、Scope控制粒度。

Go语言可用于科学计算与数值分析,但需了解其优劣。优势在于并发支持和性能,适合并行算法如分布式求解、蒙特卡洛模拟等;社区库如gonum和mat64提供基础数值计算功能;可通过cgo或接口调用C/C 、Python实现混合编程提升实用性。局限在于生态不如Python成熟,可视化和高级工具较弱,部分库文档不完善。建议结合Go特性选择合适场景并参考源码示例深入使用。
