青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品

Fork me on GitHub
隨筆 - 215  文章 - 13  trackbacks - 0
<2017年4月>
2627282930311
2345678
9101112131415
16171819202122
23242526272829
30123456


專注即時通訊及網(wǎng)游服務(wù)端編程
------------------------------------
Openresty 官方模塊
Openresty 標(biāo)準(zhǔn)模塊(Opm)
Openresty 三方模塊
------------------------------------
本博收藏大部分文章為轉(zhuǎn)載,并在文章開頭給出了原文出處,如有再轉(zhuǎn),敬請保留相關(guān)信息,這是大家對原創(chuàng)作者勞動成果的自覺尊重!!如為您帶來不便,請于本博下留言,謝謝配合。

常用鏈接

留言簿(1)

隨筆分類

隨筆檔案

相冊

Awesome

Blog

Book

GitHub

Link

搜索

  •  

積分與排名

  • 積分 - 219794
  • 排名 - 117

最新評論

閱讀排行榜

http://www.cnblogs.com/tsiangleo/p/4483657.html

1.下載并導(dǎo)入數(shù)據(jù)庫驅(qū)動包

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

然后按照里面的說明下載驅(qū)動包:

$ go get github.com/go-sql-driver/mysql

最后導(dǎo)入包即可:

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

2.連接至數(shù)據(jù)庫

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

3.執(zhí)行查詢

(1)Exec 

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

 

(2)Query 

復(fù)制代碼
rows, err := db.Query("SELECT name FROM users WHERE age = ?", age)
if err != nil {
    log.Fatal(err)
}
for rows.Next() {
    var name string
    if 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)
}
復(fù)制代碼

 

(3)QueryRow

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

 

 

(4)Prepared statements 

復(fù)制代碼
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
復(fù)制代碼

 

 

4. 事務(wù)

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

 

 

5. 各種方式效率分析

問題:db.exec和statement.exec和tx.exec的區(qū)別?

實(shí)例如下:

復(fù)制代碼
package main

import (
    "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 update
    start := 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 update
    start = 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 update
    start = 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 update
    start = 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 update
    start = 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 delete
    start := 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 delete
    start = 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 delete
    start = 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 delete
    start = 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 delete
    start = 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 query
    start := time.Now()
    rows,_ := db.Query("SELECT uid,username FROM USER")
    defer rows.Close()
    for rows.Next(){
         var name string
         var id int
        if 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 query
    start = time.Now()
    stm,_ := db.Prepare("SELECT uid,username FROM USER")
    defer stm.Close()
    rows,_ = stm.Query()
    defer rows.Close()
    for rows.Next(){
         var name string
         var id int
        if 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 query
    start = time.Now()
    tx,_ := db.Begin()
    defer tx.Commit()
    rows,_ = tx.Query("SELECT uid,username FROM USER")
    defer rows.Close()
    for rows.Next(){
         var name string
         var id int
        if 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轉(zhuǎn)string:strconv.Itoa(i)
    start := time.Now()
    for i := 1001;i<=1100;i++{
        //每次循環(huán)內(nè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 insert
    start = time.Now()
    for i := 1101;i<=1200;i++{
        //Prepare函數(shù)每次循環(huán)內(nèi)部都會去連接池獲取一個新的連接,效率低下
        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 insert
    start = time.Now()
    stm,_ := db.Prepare("INSERT INTO user(uid,username,age) values(?,?,?)")
    for i := 1201;i<=1300;i++{
        //Exec內(nèi)部并沒有去獲取連接,為什么效率還是低呢?
        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 insert
    start = time.Now()
    //Begin函數(shù)內(nèi)部會去獲取連接
    tx,_ := db.Begin()
    for i := 1301;i<=1400;i++{
        //每次循環(huán)用的都是tx內(nèi)部的連接,沒有新建連接,效率高
        tx.Exec("INSERT INTO user(uid,username,age) values(?,?,?)",i,"user"+strconv.Itoa(i),i-1000)
    }
    //最后釋放tx內(nèi)部的連接
    tx.Commit()
    
    end = time.Now()
    fmt.Println("方式4 insert total time:",end.Sub(start).Seconds())
    
    //方式5 insert
    start = time.Now()
    for i := 1401;i<=1500;i++{
        //Begin函數(shù)每次循環(huán)內(nèi)部都會去連接池獲取一個新的連接,效率低下
        tx,_ := db.Begin()
        tx.Exec("INSERT INTO user(uid,username,age) values(?,?,?)",i,"user"+strconv.Itoa(i),i-1000)
        //Commit執(zhí)行后連接也釋放了
        tx.Commit()
    }
    end = time.Now()
    fmt.Println("方式5 insert total time:",end.Sub(start).Seconds())
}
復(fù)制代碼

程序輸出結(jié)果:

方式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. 深入內(nèi)部分析原因分析

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

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

實(shí)現(xiàn):open函數(shù)并沒有創(chuàng)建連接,它只是驗(yàn)證參數(shù)是否合法。然后開啟一個單獨(dú)goroutines去監(jiān)聽是否需要建立新的連接,當(dāng)有請求建立新連接時就創(chuàng)建新連接。

注意:open函數(shù)應(yīng)該被調(diào)用一次,通常是沒必要close的。

 

(2)DB.Exec()

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

實(shí)現(xiàn):DB交給內(nèi)部的exec方法負(fù)責(zé)查詢。exec會首先調(diào)用DB內(nèi)部的conn方法從連接池里面獲得一個連接。然后檢查內(nèi)部的driver.Conn實(shí)現(xiàn)了Execer接口沒有,如果實(shí)現(xiàn)了該接口,會調(diào)用Execer接口的Exec方法執(zhí)行查詢;否則調(diào)用Conn接口的Prepare方法負(fù)責(zé)查詢。

 

(3)DB.Query()

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

實(shí)現(xiàn):DB交給內(nèi)部的query方法負(fù)責(zé)查詢。query首先調(diào)用DB內(nèi)部的conn方法從連接池里面獲得一個連接,然后調(diào)用內(nèi)部的queryConn方法負(fù)責(zé)查詢。

 

(4)DB.QueryRow()

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

實(shí)現(xiàn):轉(zhuǎn)交給DB.Query()查詢

 

(5)db.Prepare()

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

實(shí)現(xiàn):DB交給內(nèi)部的prepare方法負(fù)責(zé)查詢。prepare首先調(diào)用DB內(nèi)部的conn方法從連接池里面獲得一個連接,然后調(diào)用driverConn的prepareLocked方法負(fù)責(zé)查詢。

Stmt相關(guān)方法:

st.Exec()

st.Query()

st.QueryRow()

st.Close()

 

(6)db.Begin()

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

實(shí)現(xiàn):DB交給內(nèi)部的begin方法負(fù)責(zé)處理。begin首先調(diào)用DB內(nèi)部的conn方法從連接池里面獲得一個連接,然后調(diào)用Conn接口的Begin方法獲得一個TX。

TX相關(guān)方法:

//內(nèi)部執(zhí)行流程和上面那些差不多,只是沒有先去獲取連接的一步,因?yàn)檫@些操作是和TX關(guān)聯(lián)的,Tx建立的時候就和一個連接綁定了,所以這些操作內(nèi)部共用一個TX內(nèi)部的連接。

tx.Exec() 

tx.Query()

tx.QueryRow()

tx.Prepare()

tx.Commit()

tx.Rollback()

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

例子:

復(fù)制代碼
  updateMoney, err := db.Prepare("UPDATE balance SET money=money+? WHERE id=?")

  ...

  tx, err := db.Begin()

  ...

  res, err := tx.Stmt(updateMoney).Exec(123.45, 98293203)
復(fù)制代碼

 

 (7)源碼中Stmt的定義

復(fù)制代碼
// 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   *Tx

         txsi *driverStmt

 

         mu     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

}
復(fù)制代碼

 

(7)幾個主要struct的內(nèi)部主要的數(shù)據(jù)結(jié)構(gòu)

 

參考資料

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

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

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

posted on 2017-04-18 14:11 思月行云 閱讀(934) 評論(1)  編輯 收藏 引用 所屬分類: Golang

FeedBack:
# re: Golang操作MySql數(shù)據(jù)庫 2017-04-18 16:29 思月行云
select多個字段時的寫法:

//查詢數(shù)據(jù)
rows, err := db.Query("SELECT * FROM userinfo")
checkErr(err)

for rows.Next() {
var uid int
var username string
var department string
var created string
err = rows.Scan(&uid, &username, &department, &created)
checkErr(err)
fmt.Println(uid)
fmt.Println(username)
fmt.Println(department)
fmt.Println(created)
}   回復(fù)  更多評論
  
青青草原综合久久大伊人导航_色综合久久天天综合_日日噜噜夜夜狠狠久久丁香五月_热久久这里只有精品
  • <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
      <noscript id="pjuwb"></noscript>
            <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
              <dd id="pjuwb"></dd>
              <abbr id="pjuwb"></abbr>
              欧美一区激情视频在线观看| 裸体歌舞表演一区二区| 亚洲另类在线一区| 欧美精品日韩三级| 亚洲一区二区三区四区五区午夜| 亚洲三级网站| 欧美日本中文字幕| 亚洲欧美日韩精品在线| 香蕉久久一区二区不卡无毒影院 | 免费在线观看日韩欧美| 亚洲美女在线一区| 亚洲一级黄色av| 黄色一区三区| 亚洲精选成人| 国产亚洲一区二区三区在线观看 | 久久综合久久美利坚合众国| 亚洲人成在线观看一区二区| 一区二区三区三区在线| 韩日精品中文字幕| 亚洲伦理自拍| 国产专区综合网| 亚洲精选一区| 狠狠色伊人亚洲综合网站色| 亚洲精品乱码视频| 夜夜精品视频| 久久久九九九九| 一本色道88久久加勒比精品| 亚洲欧美日韩电影| 亚洲区一区二| 久久激情中文| 亚洲一区二区动漫| 免费短视频成人日韩| 香港久久久电影| 欧美精品日韩一区| 蜜臀va亚洲va欧美va天堂| 国产精品国产a| 亚洲国产高清一区二区三区| 国产免费观看久久黄| 亚洲精品国产系列| 亚洲国产视频一区二区| 欧美一区二区三区四区在线| 宅男精品导航| 欧美成人dvd在线视频| 久久综合久久综合久久综合| 国产精品青草久久久久福利99| 亚洲欧洲精品一区二区三区| 韩国精品在线观看| 午夜精品久久久久久久99水蜜桃 | 亚洲卡通欧美制服中文| 亚洲国产成人久久综合| 欧美在线视频免费| 欧美一区二区黄| 欧美午夜无遮挡| 日韩网站在线观看| 一本大道久久a久久精二百| 欧美α欧美αv大片| 麻豆av一区二区三区| 国产主播精品| 久久久精品一品道一区| 久久一区二区三区四区| 狠狠色狠色综合曰曰| 欧美在线影院| 久久最新视频| 亚洲大胆美女视频| 美女网站久久| 亚洲黄网站在线观看| 日韩一区二区免费高清| 欧美日产国产成人免费图片| 99re亚洲国产精品| 亚洲影院色在线观看免费| 欧美日韩中文字幕在线| 亚洲图片欧美午夜| 久久国产福利国产秒拍| 国产综合欧美在线看| 久久久久久色| 亚洲激情电影在线| 亚洲一区二区三区高清 | 欧美精品久久天天躁| 日韩一区二区精品| 午夜伦欧美伦电影理论片| 国产九色精品成人porny| 久久xxxx| 亚洲精品国产精品国自产观看浪潮| 一本色道久久88综合日韩精品| 欧美日在线观看| 欧美一进一出视频| 欧美成人免费在线| 国产精品看片资源| 亚洲国产一区二区三区青草影视| 一区二区三区你懂的| 国产精品最新自拍| 欧美成在线视频| 亚洲无限乱码一二三四麻| 久久人人爽人人| 日韩一级视频免费观看在线| 国产美女一区二区| 久久综合电影| 亚洲性人人天天夜夜摸| 欧美jizzhd精品欧美巨大免费| 一区二区三区毛片| 国内免费精品永久在线视频| 欧美片第1页综合| 久久爱www.| 一区二区三区国产在线观看| 毛片av中文字幕一区二区| 亚洲视频中文字幕| 亚洲国产高清一区| 国产视频亚洲| 欧美视频免费看| 麻豆精品一区二区综合av | 久久精品一区二区国产| 99国产欧美久久久精品| 黑人操亚洲美女惩罚| 欧美视频免费在线| 欧美黄色免费| 久久夜色精品国产欧美乱极品| 在线视频中文亚洲| 亚洲国产精品黑人久久久| 久久久久国内| 欧美一级二级三级蜜桃| 亚洲视频在线二区| 亚洲美女淫视频| 91久久精品一区二区三区| 国产亚洲美州欧州综合国| 欧美日韩理论| 欧美日韩高清在线播放| 久久天天躁狠狠躁夜夜爽蜜月| 香蕉乱码成人久久天堂爱免费| 一区二区三区四区蜜桃| 亚洲剧情一区二区| 亚洲免费观看| 亚洲精品国产欧美| 亚洲精品少妇网址| 亚洲精品乱码久久久久久日本蜜臀 | 一本色道久久加勒比88综合| 亚洲国产日韩一级| 欧美国产精品v| 欧美成人午夜激情在线| 另类激情亚洲| 蜜臀久久久99精品久久久久久 | 久久天天躁狠狠躁夜夜av| 久久国产福利| 久久亚洲精品中文字幕冲田杏梨| 久久精品天堂| 久久躁日日躁aaaaxxxx| 狼狼综合久久久久综合网| 久热精品视频在线观看| 老司机午夜精品视频| 免费在线成人av| 欧美韩日一区二区三区| 亚洲国产婷婷香蕉久久久久久| 欧美黑人在线播放| 亚洲国产婷婷| 一区二区久久久久| 亚洲一区二区三区免费在线观看| 久久精品亚洲精品| 99精品99| 亚洲淫片在线视频| 久久精品国产999大香线蕉| 久久久精品动漫| 欧美成人午夜激情| 99re6这里只有精品视频在线观看| 亚洲图片在区色| 久久久水蜜桃av免费网站| 欧美成人国产一区二区| 欧美性猛交xxxx乱大交退制版| 国产日韩欧美成人| 亚洲黄色在线视频| 亚洲欧美日韩区| 美女视频黄免费的久久| 亚洲精品在线观| 午夜在线观看欧美| 欧美国产日本| 国产婷婷97碰碰久久人人蜜臀| 亚洲电影av| 亚洲欧美日韩精品综合在线观看| 久久先锋影音| 一本色道久久加勒比88综合| 久久精品日产第一区二区三区| 欧美日产在线观看| 激情综合自拍| 午夜精品一区二区在线观看| 美女久久一区| 亚洲一区二区免费看| 女仆av观看一区| 国产精品私人影院| 亚洲免费观看| 久久一综合视频| 亚洲午夜未删减在线观看| 麻豆精品精华液| 国产色产综合色产在线视频| 在线视频中文亚洲| 欧美99久久| 久久国产黑丝| 国产精品日韩精品| 亚洲视频欧美视频| 亚洲国产成人av| 久久婷婷激情| 韩日精品视频一区| 久久成人精品视频|