使用 go-sqlmock 並將參數插入模擬查詢的問題
在PHP开发中,数据库操作是非常常见的需求,而对于一些需要测试的场景,我们又不希望直接操作真实的数据库。这时候,我们可以使用go-sqlmock来模拟数据库查询,从而达到我们想要的测试效果。本文将向大家介绍如何使用go-sqlmock,并将参数插入模拟查询的问题。无论你是PHP开发初学者还是有一定经验的开发者,通过学习本文,你都能够轻松掌握这一技巧,提升你的开发效率。
问题内容
我正在尝试使用 go-sqlmock 模拟我的查询函数并类似地复制数据库表。但是,我没有得到我期望的结果。查询的行为不正常,参数没有插入到查询中并且实际结果不正确。我在这里做错了什么?
这是我正在嘲笑的函数和查询:
func (y *yumdatabase) gettransactionid(pkg string) (int, error) { var id int queryfortid := "select tid from trans_cmdline where cmdline like '%install " + pkg + "%' order by tid desc limit 1" row := y.db.queryrow(queryfortid) switch err := row.scan(&id); err { case sql.errnorows: fmt.println("no rows were returned") return 0, err case nil: return id, nil default: return 0, err } }
这是模拟测试功能:
func testgettransactionid(t *testing.t) { db, mock, err := sqlmock.new() if err != nil { t.fatalf("err not expected: %v", err) } pkg := "tcpdump" rows := sqlmock.newrows([]string{"tid"}).addrow("1").addrow("3") mock.expectquery("select tid from trans_cmdline where cmdline like '%install " + pkg + "%' order by tid desc limit 1").willreturnrows(rows) mockdb := &yumdatabase{ db: db, } got, err := mockdb.gettransactionid("tcpdump") assert.equal(t, 3, got) }
如果上述工作按预期进行,我会在“got”中返回“3”,但我会返回“1”
其次,是否可以将 rows 更改为以下内容:
rows := sqlmock.newrows([]string{"tid", "cmdline"}).addrow("1", "install test").addrow("3", "delete test2")
实际上进行了比较“where cmdline like '%install xyz%'”,因为我尝试了这个,并且收到了以下错误(所有主要代码都构建并工作,包括查询,所以这是一个问题我猜是我写的模拟代码):
error sql: expected 2 destination arguments in Scan, not 1
我希望看到从 sql 查询返回的最高 tid,而不是“addrow”中指定的第一个 tid,并且我希望查询实现对模拟中的“cmdline”行的检查。
解决方法
我就是这样管理你的需求的。首先,让我分享代码,然后,我将引导您完成所有相关更改。该代码包含在两个文件中:repo.go
和 repo_test.go
。
repo.go
文件
package repo import ( "database/sql" "fmt" ) func gettransactionid(db *sql.db, pkg string) (int, error) { var id int row := db.queryrow("select tid from trans_cmdline where cmdline like '%install $1%' order by tid desc limit 1", pkg) switch err := row.scan(&id); err { case sql.errnorows: fmt.println("no rows were returned") return 0, err case nil: return id, nil default: return 0, err } }
这里有两个小改进:
*sql.db
作为参数传入。正如最佳实践所建议的,函数是一等公民。这就是为什么我更愿意尽可能坚持使用它们。- 我使用准备好的语句来传递查询的参数。不是简单的字符串连接。因此,可以更轻松地拦截传递给查询的参数并对其设置期望
现在让我们切换到测试代码。
repo_test.go
文件
package repo import ( "database/sql" "testing" "github.com/DATA-DOG/go-sqlmock" "github.com/stretchr/testify/assert" ) func TestGetTransactionId(t *testing.T) { db, mock, err := sqlmock.New(sqlmock.QueryMatcherOption(sqlmock.QueryMatcherEqual)) if err != nil { t.Fatalf("err not expected while opening mock db, %v", err) } t.Run("HappyPath", func(t *testing.T) { rows := sqlmock.NewRows([]string{"tid"}).AddRow("1") mock.ExpectQuery("SELECT tid FROM trans_cmdline WHERE cmdline LIKE '%install $1%' ORDER BY tid DESC LIMIT 1"). WithArgs("tcpdump"). WillReturnRows(rows) got, err := GetTransactionId(db, "tcpdump") assert.Equal(t, 1, got) assert.Nil(t, err) }) t.Run("NoRowsReturned", func(t *testing.T) { mock.ExpectQuery("SELECT tid FROM trans_cmdline WHERE cmdline LIKE '%install $1%' ORDER BY tid DESC LIMIT 1"). WithArgs("tcpdump"). WillReturnError(sql.ErrNoRows) got, err := GetTransactionId(db, "tcpdump") assert.Equal(t, 0, got) assert.Equal(t, sql.ErrNoRows, err) }) }
这里,您需要注意更多更改:
- 在实例化
db
和mock
时,您应该将sqlmock.querymatcherequal
作为参数传递给。因此,它将完全匹配查询。 -
expectquery
方法现在使用准备好的语句功能并需要一个参数(例如本例中的tcpdump
)。 - 重构了断言以利用
github.com/stretchr/testify/assert
包。
我希望这可以帮助您解决问题,请告诉我!
以上是使用 go-sqlmock 並將參數插入模擬查詢的問題的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

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

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

Go語言在構建高效且可擴展的系統中表現出色,其優勢包括:1.高性能:編譯成機器碼,運行速度快;2.並發編程:通過goroutines和channels簡化多任務處理;3.簡潔性:語法簡潔,降低學習和維護成本;4.跨平台:支持跨平台編譯,方便部署。

Golang在並發性上優於C ,而C 在原始速度上優於Golang。 1)Golang通過goroutine和channel實現高效並發,適合處理大量並發任務。 2)C 通過編譯器優化和標準庫,提供接近硬件的高性能,適合需要極致優化的應用。

Golang和Python各有优势:Golang适合高性能和并发编程,Python适用于数据科学和Web开发。Golang以其并发模型和高效性能著称,Python则以简洁语法和丰富库生态系统著称。

Golang在性能和可擴展性方面優於Python。 1)Golang的編譯型特性和高效並發模型使其在高並發場景下表現出色。 2)Python作為解釋型語言,執行速度較慢,但通過工具如Cython可優化性能。

goimpactsdevelopmentpositationality throughspeed,效率和模擬性。 1)速度:gocompilesquicklyandrunseff,IdealforlargeProjects.2)效率:效率:ITScomprehenSevestAndardArdardArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdArdEcceSteral Depentencies,增強的Depleflovelmentimency.3)簡單性。

C 更適合需要直接控制硬件資源和高性能優化的場景,而Golang更適合需要快速開發和高並發處理的場景。 1.C 的優勢在於其接近硬件的特性和高度的優化能力,適合遊戲開發等高性能需求。 2.Golang的優勢在於其簡潔的語法和天然的並發支持,適合高並發服務開發。

Golang和C 在性能競賽中的表現各有優勢:1)Golang適合高並發和快速開發,2)C 提供更高性能和細粒度控制。選擇應基於項目需求和團隊技術棧。

Golang和C 在性能上的差異主要體現在內存管理、編譯優化和運行時效率等方面。 1)Golang的垃圾回收機制方便但可能影響性能,2)C 的手動內存管理和編譯器優化在遞歸計算中表現更為高效。
