This article uses database transactions based on the github.com/jmoiron/sqlx package introduced in the previous section when accessing the database.
defer
Before using database transactions, you need to understand the GO language’s defer keyword. Defer is a deferred execution statement in GO. The statements following defer will be deferred by GO, and when the function is about to end, the statements following defer will be executed in reverse order. That is, the statements that defer first execute last. Defer looks a lot like the finally statement in Java or C#. Let’s take a look at defer with an example.
Package main import "FMT" func main() {fmt.println (" start to defer ") defer fmt.println (1) defer fmt.println (2) defer Println(3) panic(" test panic") FMT.Println("defer end ")}Copy the code
Code running results:
1 panic: Test Panic goroutine 1 [running]: main.main() /tests/gookokok/main.go:10 +0x1b8 Process finished with exit code 2Copy the code
You can see the reverse output of defer by running the results. Later, a panic was manually triggered, affecting the last line of output to end defer. Go performed defer first when panic was triggered, proving that defer was safe enough to be used to mutually exclusive unlock, close files, or close database transactions.
The transaction
SQLX extends methods such as MustBegin(), MustExec(), etc. using transactions compared to Database/SQL, so you don’t need to handle many errors manually in your code. MustBegin triggers panic when an error occurs, rather than returning an error, so that errors can be handled uniformly at a higher level in the code.
tx, err := db.Begin()
err = tx.Exec(...)
err = tx.Commit()
tx := db.MustBegin()
tx.MustExec(...)
err = tx.Commit()
Copy the code
SQLX supports both methods of starting transactions. MustBegin returns SQLx. Tx, which also provides APIS such as Select and Get. Performing database operations is the same as using SQLX.db.
The above code executes tx.mustexec (…) If an error occurs, the code will not have a chance to run to tx.mit (), so the database connection will not be closed until go does garbage collection, and high concurrency may fill up the database connection count, causing the site to become inaccessible.
tx := db.MustBegin()
defer tx.Rollback()
tx.MustExec(...)
err = tx.Commit()
Copy the code
Add defer tx.rollback () to the code to solve the problem. From the previous introduction, we know that defer executes at the end of the method, so that even code exceptions do not affect the database connection.
Golang Web Development based on Gin: Using database transactions