日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 运维知识 > 数据库 >内容正文

数据库

golang学习之旅:使用go语言操作mysql数据库(自己测试了)

發布時間:2024/9/20 数据库 24 豆豆
生活随笔 收集整理的這篇文章主要介紹了 golang学习之旅:使用go语言操作mysql数据库(自己测试了) 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

1.下載并導入數據庫驅動包

官方不提供實現,先下載第三方的實現,點擊這里查看各種各樣的實現版本。
這里選擇了Go-MySQL-Driver這個實現。地址是:https://github.com/go-sql-driver/mysql/。

然后按照里面的說明下載驅動包:

$ go get github.com/go-sql-driver/mysql(個人感覺是讓這個地址生效的語句)

最后導入包即可:

import "database/sql" import _ "github.com/go-sql-driver/mysql"

2.連接至數據庫

db, err := sql.Open("mysql", "root:root@/uestcbook")

3.執行查詢

(1)Exec?

result, err := db.Exec("INSERT INTO users (name, age) VALUES (?, ?)","gopher",27, )

?

(2)Query?

rows, err := db.Query("SELECT name FROM users WHERE age = ?", age) if err != nil {log.Fatal(err) } for rows.Next() {var name stringif err := rows.Scan(&name); err != nil {log.Fatal(err)}fmt.Printf("%s is %d\n", name, age) } if err := rows.Err(); err != nil {log.Fatal(err) }

?

(3)QueryRow

var age int64 row := db.QueryRow("SELECT age FROM users WHERE name = ?", name) err := row.Scan(&age)

?

?

(4)Prepared statements?

age := 27 stmt, err := db.Prepare("SELECT name FROM users WHERE age = ?") if err != nil {log.Fatal(err) } rows, err := stmt.Query(age) // process rows

?

?

4. 事務

tx, err := db.Begin() if err != nil {log.Fatal(err) }

?

?

5. 各種方式效率分析

問題:db.exec和statement.exec和tx.exec的區別?

實例如下:

package mainimport ("strconv""database/sql"_ "github.com/go-sql-driver/mysql""fmt""time""log" )var db = &sql.DB{}func init(){db,_ = sql.Open("mysql", "root:root@/book") } func main() {insert()query()update()query()delete() }func update(){//方式1 updatestart := time.Now()for i := 1001;i<=1100;i++{db.Exec("UPdate user set age=? where uid=? ",i,i)}end := time.Now()fmt.Println("方式1 update total time:",end.Sub(start).Seconds())//方式2 updatestart = time.Now()for i := 1101;i<=1200;i++{stm,_ := db.Prepare("UPdate user set age=? where uid=? ")stm.Exec(i,i)stm.Close()}end = time.Now()fmt.Println("方式2 update total time:",end.Sub(start).Seconds())//方式3 updatestart = time.Now()stm,_ := db.Prepare("UPdate user set age=? where uid=?")for i := 1201;i<=1300;i++{stm.Exec(i,i)}stm.Close()end = time.Now()fmt.Println("方式3 update total time:",end.Sub(start).Seconds())//方式4 updatestart = time.Now()tx,_ := db.Begin()for i := 1301;i<=1400;i++{tx.Exec("UPdate user set age=? where uid=?",i,i)}tx.Commit()end = time.Now()fmt.Println("方式4 update total time:",end.Sub(start).Seconds())//方式5 updatestart = time.Now()for i := 1401;i<=1500;i++{tx,_ := db.Begin()tx.Exec("UPdate user set age=? where uid=?",i,i)tx.Commit()}end = time.Now()fmt.Println("方式5 update total time:",end.Sub(start).Seconds())}func delete(){//方式1 deletestart := time.Now()for i := 1001;i<=1100;i++{db.Exec("DELETE FROM USER WHERE uid=?",i)}end := time.Now()fmt.Println("方式1 delete total time:",end.Sub(start).Seconds())//方式2 deletestart = time.Now()for i := 1101;i<=1200;i++{stm,_ := db.Prepare("DELETE FROM USER WHERE uid=?")stm.Exec(i)stm.Close()}end = time.Now()fmt.Println("方式2 delete total time:",end.Sub(start).Seconds())//方式3 deletestart = time.Now()stm,_ := db.Prepare("DELETE FROM USER WHERE uid=?")for i := 1201;i<=1300;i++{stm.Exec(i)}stm.Close()end = time.Now()fmt.Println("方式3 delete total time:",end.Sub(start).Seconds())//方式4 deletestart = time.Now()tx,_ := db.Begin()for i := 1301;i<=1400;i++{tx.Exec("DELETE FROM USER WHERE uid=?",i)}tx.Commit()end = time.Now()fmt.Println("方式4 delete total time:",end.Sub(start).Seconds())//方式5 deletestart = time.Now()for i := 1401;i<=1500;i++{tx,_ := db.Begin()tx.Exec("DELETE FROM USER WHERE uid=?",i)tx.Commit()}end = time.Now()fmt.Println("方式5 delete total time:",end.Sub(start).Seconds())}func query(){//方式1 querystart := time.Now()rows,_ := db.Query("SELECT uid,username FROM USER")defer rows.Close()for rows.Next(){var name stringvar id intif err := rows.Scan(&id,&name); err != nil {log.Fatal(err)}//fmt.Printf("name:%s ,id:is %d\n", name, id) }end := time.Now()fmt.Println("方式1 query total time:",end.Sub(start).Seconds())//方式2 querystart = time.Now()stm,_ := db.Prepare("SELECT uid,username FROM USER")defer stm.Close()rows,_ = stm.Query()defer rows.Close()for rows.Next(){var name stringvar id intif err := rows.Scan(&id,&name); err != nil {log.Fatal(err)}// fmt.Printf("name:%s ,id:is %d\n", name, id) }end = time.Now()fmt.Println("方式2 query total time:",end.Sub(start).Seconds())//方式3 querystart = time.Now()tx,_ := db.Begin()defer tx.Commit()rows,_ = tx.Query("SELECT uid,username FROM USER")defer rows.Close()for rows.Next(){var name stringvar id intif err := rows.Scan(&id,&name); err != nil {log.Fatal(err)}//fmt.Printf("name:%s ,id:is %d\n", name, id) }end = time.Now()fmt.Println("方式3 query total time:",end.Sub(start).Seconds()) }func insert() {//方式1 insert//strconv,int轉string:strconv.Itoa(i)start := time.Now()for i := 1001;i<=1100;i++{//每次循環內部都會去連接池獲取一個新的連接,效率低下db.Exec("INSERT INTO user(uid,username,age) values(?,?,?)",i,"user"+strconv.Itoa(i),i-1000)}end := time.Now()fmt.Println("方式1 insert total time:",end.Sub(start).Seconds())//方式2 insertstart = time.Now()for i := 1101;i<=1200;i++{//Prepare函數每次循環內部都會去連接池獲取一個新的連接,效率低下stm,_ := db.Prepare("INSERT INTO user(uid,username,age) values(?,?,?)")stm.Exec(i,"user"+strconv.Itoa(i),i-1000)stm.Close()}end = time.Now()fmt.Println("方式2 insert total time:",end.Sub(start).Seconds())//方式3 insertstart = time.Now()stm,_ := db.Prepare("INSERT INTO user(uid,username,age) values(?,?,?)")for i := 1201;i<=1300;i++{//Exec內部并沒有去獲取連接,為什么效率還是低呢?stm.Exec(i,"user"+strconv.Itoa(i),i-1000)}stm.Close()end = time.Now()fmt.Println("方式3 insert total time:",end.Sub(start).Seconds())//方式4 insertstart = time.Now()//Begin函數內部會去獲取連接tx,_ := db.Begin()for i := 1301;i<=1400;i++{//每次循環用的都是tx內部的連接,沒有新建連接,效率高tx.Exec("INSERT INTO user(uid,username,age) values(?,?,?)",i,"user"+strconv.Itoa(i),i-1000)}//最后釋放tx內部的連接 tx.Commit()end = time.Now()fmt.Println("方式4 insert total time:",end.Sub(start).Seconds())//方式5 insertstart = time.Now()for i := 1401;i<=1500;i++{//Begin函數每次循環內部都會去連接池獲取一個新的連接,效率低下tx,_ := db.Begin()tx.Exec("INSERT INTO user(uid,username,age) values(?,?,?)",i,"user"+strconv.Itoa(i),i-1000)//Commit執行后連接也釋放了 tx.Commit()}end = time.Now()fmt.Println("方式5 insert total time:",end.Sub(start).Seconds()) }

程序輸出結果:

方式1 insert total time: 3.7952171
方式2 insert total time: 4.3162468
方式3 insert total time: 4.3392482
方式4 insert total time: 0.3970227
方式5 insert total time: 7.3894226
方式1 query total time: 0.0070004
方式2 query total time: 0.0100006
方式3 query total time: 0.0100006
方式1 update total time: 7.3394198
方式2 update total time: 7.8464488
方式3 update total time: 6.0053435
方式4 update total time: 0.6630379000000001
方式5 update total time: 4.5402597
方式1 query total time: 0.0070004
方式2 query total time: 0.0060004
方式3 query total time: 0.008000400000000001
方式1 delete total time: 3.8652211000000003
方式2 delete total time: 3.8582207
方式3 delete total time: 3.6972114
方式4 delete total time: 0.43202470000000004
方式5 delete total time: 3.7972172

?

6. 深入內部分析原因分析

(1)sql.Open("mysql", "username:pwd@/databasename")

功能:返回一個DB對象,DB對象對于多個goroutines并發使用是安全的,DB對象內部封裝了連接池。

實現:open函數并沒有創建連接,它只是驗證參數是否合法。然后開啟一個單獨goroutines去監聽是否需要建立新的連接,當有請求建立新連接時就創建新連接。

注意:open函數應該被調用一次,通常是沒必要close的。

?

(2)DB.Exec()

功能:執行不返回行(row)的查詢,比如INSERT,UPDATE,DELETE

實現:DB交給內部的exec方法負責查詢。exec會首先調用DB內部的conn方法從連接池里面獲得一個連接。然后檢查內部的driver.Conn實現了Execer接口沒有,如果實現了該接口,會調用Execer接口的Exec方法執行查詢;否則調用Conn接口的Prepare方法負責查詢。

?

(3)DB.Query()

功能:用于檢索(retrieval),比如SELECT

實現:DB交給內部的query方法負責查詢。query首先調用DB內部的conn方法從連接池里面獲得一個連接,然后調用內部的queryConn方法負責查詢。

?

(4)DB.QueryRow()

功能:用于返回單行的查詢

實現:轉交給DB.Query()查詢

?

(5)db.Prepare()

功能:返回一個Stmt。Stmt對象可以執行Exec,Query,QueryRow等操作。

實現:DB交給內部的prepare方法負責查詢。prepare首先調用DB內部的conn方法從連接池里面獲得一個連接,然后調用driverConn的prepareLocked方法負責查詢。

Stmt相關方法:

st.Exec()

st.Query()

st.QueryRow()

st.Close()

?

(6)db.Begin()

功能:開啟事務,返回Tx對象。調用該方法后,這個TX就和指定的連接綁定在一起了。一旦事務提交或者回滾,該事務綁定的連接就還給DB的連接池。

實現:DB交給內部的begin方法負責處理。begin首先調用DB內部的conn方法從連接池里面獲得一個連接,然后調用Conn接口的Begin方法獲得一個TX。

TX相關方法:

//內部執行流程和上面那些差不多,只是沒有先去獲取連接的一步,因為這些操作是和TX關聯的,Tx建立的時候就和一個連接綁定了,所以這些操作內部共用一個TX內部的連接。

tx.Exec()?

tx.Query()

tx.QueryRow()

tx.Prepare()

tx.Commit()

tx.Rollback()

tx.Stmt()//用于將一個已存在的statement和tx綁定在一起。一個statement可以不和tx關聯,比如db.Prepare()返回的statement就沒有和TX關聯。

例子:

updateMoney, err := db.Prepare("UPDATE balance SET money=money+? WHERE id=?")...tx, err := db.Begin()...res, err := tx.Stmt(updateMoney).Exec(123.45, 98293203)

?

?(7)源碼中Stmt的定義

// Stmt is a prepared statement. Stmt is safe for concurrent use by multiple goroutines. type Stmt struct {// Immutable: db *DB // where we came from query string // that created the Stmt stickyErr error // if non-nil, this error is returned for all operations closemu sync.RWMutex // held exclusively during close, for read otherwise.// If in a transaction, else both nil: tx *Txtxsi *driverStmtmu sync.Mutex // protects the rest of the fields closed bool// css is a list of underlying driver statement interfaces// that are valid on particular connections. This is only// used if tx == nil and one is found that has idle// connections. If tx != nil, txsi is always used. css []connStmt}

?

(7)幾個主要struct的內部主要的數據結構

?

參考資料

https://github.com/golang/go/wiki/SQLInterface

https://github.com/go-sql-driver/mysql/

http://golang.org/pkg/database/sql/


來源:https://www.cnblogs.com/tsiangleo/p/4483657.html

總結

以上是生活随笔為你收集整理的golang学习之旅:使用go语言操作mysql数据库(自己测试了)的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 亚洲av成人精品一区二区三区在线播放 | 免费av软件 | 午夜成年人 | 毛片网在线观看 | 激情欧美综合 | 精品久久久无码中文字幕边打电话 | 国产又粗又猛又爽又黄的网站 | 免费麻豆国产一区二区三区四区 | 国产一级一片免费播放 | 波多野结衣黄色网址 | 亚洲天堂av网 | 国产真实乱偷精品视频 | 日批视频在线播放 | 国产午夜电影在线观看 | 亚洲乱码国产乱码精品精软件 | 不卡一区二区在线视频 | 黄色一级片久久 | 日韩成人高清在线 | 国产日韩欧美成人 | 老熟妇精品一区二区三区 | 国产亚洲精品av | 日韩久久久久久久久久久 | 中文字幕免费在线看线人动作大片 | 麻豆视频免费观看 | 逼特逼在线视频 | 在线免费看污片 | 豆花免费跳转入口官网 | 欧美日韩一级二级 | 青青草视频播放器 | 爽插| 69天堂网| 一级黄毛片 | 调教撅屁股啪调教打臀缝av | 国产亚洲精品aaaaaaa片 | 精品乱码久久久久久中文字幕 | 精品国产一区二区三区四区阿崩 | 91在线网站 | 久久久精品 | 俄罗斯毛片基地 | 精品人妻无码一区二区三区换脸 | 69免费视频| 日韩美女视频网站 | 亚洲AV无码成人精品国产一区 | 538国产精品视频一区二区 | 男人天堂久久久 | 成人网导航 | 日韩精品一区不卡 | 红猫大本营在线观看的 | 一区二区三区视频观看 | 色涩综合| 中文字幕亚洲乱码熟女一区二区 | 无码精品在线观看 | 日本韩国欧美一区二区三区 | 强制高潮抽搐哭叫求饶h | 粉嫩av一区二区三区四区五区 | av午夜激情 | 毛片久久久久久 | 深夜的私人秘书 | 国产精品亚洲一区 | 在线免费观看黄色av | 色天天综合 | 无码久久精品国产亚洲av影片 | 粉豆av| 国产91页 | 人日人视频 | 黄色片免费视频 | 娇小萝被两个黑人用半米长 | 日本高清视频www | 国产中文字幕亚洲 | 国产美女无遮挡免费视频 | 深田咏美av在线 | 少妇饥渴难耐 | 国产精品国产三级国产传播 | 国产特级视频 | 国产av不卡一区 | www国产亚洲精品久久麻豆 | 欧美日韩午夜激情 | 日日麻批免费视频播放 | 亚洲第一页中文字幕 | 国产女在线 | 欧美美女在线 | 无码人妻熟妇av又粗又大 | 天堂色区 | 四虎音影| 爱爱免费网址 | 精品96久久久久久中文字幕无 | 亚洲女同一区二区 | 欲色综合 | 久久国产精品系列 | 自拍偷拍日韩 | 青青草国产在线视频 | 国产一区观看 | 肉色超薄丝袜脚交69xx | 日本精品久久久久 | 亚洲美女操| 亚洲成人xxx | 欧美乱妇日本无乱码特黄大片 | 亚洲第一区在线播放 | 亚洲av无码一区东京热久久 |