成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

Golang如何優(yōu)雅連接MySQL數(shù)據(jù)庫(kù)?

數(shù)據(jù)庫(kù) MySQL
Go原生就支持連接數(shù)據(jù)庫(kù),所以在使用 Golang 開(kāi)發(fā)時(shí),當(dāng)需要數(shù)據(jù)庫(kù)交互時(shí),即可使用database/sql包。

 Go原生就支持連接數(shù)據(jù)庫(kù),所以在使用 Golang 開(kāi)發(fā)時(shí),當(dāng)需要數(shù)據(jù)庫(kù)交互時(shí),即可使用database/sql包。


在Go中訪問(wèn)DB需用sql.DB接口:可創(chuàng)建語(yǔ)句(statement)和事務(wù)(transaction),執(zhí)行查詢,獲取結(jié)果。

使用DB時(shí),除database/sql包,還需引入想使用的特定DB驅(qū)動(dòng)。官方不提供實(shí)現(xiàn),需要先下載三方實(shí)現(xiàn),點(diǎn)擊這里查看各種各樣的實(shí)現(xiàn)版本。

通常DB選型MySQL,所以選型驅(qū)動(dòng)為:github.com/go-sql-driver/mysql,需引入包:

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

 包名前的"_" 

import 下劃線(如:import _ github/demo)的作用:當(dāng)導(dǎo)入一個(gè)包時(shí),該包下的文件里所有init()函數(shù)都會(huì)被執(zhí)行,然而,有些時(shí)候我們并不需要把整個(gè)包都導(dǎo)入進(jìn)來(lái),僅僅是是希望它執(zhí)行init()函數(shù)而已。這個(gè)時(shí)候就可以使用 import _ 引用該包。

上面的MySQL驅(qū)動(dòng)中引入的就是MySQL包中各個(gè)init()方法,你無(wú)法通過(guò)包名來(lái)調(diào)用包中的其他函數(shù)。導(dǎo)入時(shí),驅(qū)動(dòng)的初始化函數(shù)會(huì)調(diào)用sql.Register將自己注冊(cè)在database/sql包的全局變量sql.drivers中,以便以后通過(guò)sql.Open訪問(wèn)。

 案例用數(shù)據(jù)表

 初始化數(shù)據(jù)庫(kù)連接[[353659]] 

 sql.Open()中的數(shù)據(jù)庫(kù)連接串格式為:"用戶名:密碼@tcp(IP:端口)/數(shù)據(jù)庫(kù)?charset=utf8"。DB的類型為:*sql.DB,有DB后即可執(zhí)行CRUD。

Go將數(shù)據(jù)庫(kù)操作分為兩類:Query與Exec

  • Query表示查詢,它會(huì)從數(shù)據(jù)庫(kù)獲取查詢結(jié)果(一系列行,可能為空)。
  • Exec表示執(zhí)行語(yǔ)句,它不會(huì)返回行。

常見(jiàn)數(shù)據(jù)庫(kù)操作模式:

  • QueryRow只返回一行的查詢,作為Query的一個(gè)常見(jiàn)特例。
  • Prepare準(zhǔn)備一個(gè)需要多次使用的語(yǔ)句,供后續(xù)執(zhí)行用。

查詢操作

  1. var user User 
  2. rows, e := DB.Query("select * from user where id in (1,2,3)"
  3. if e == nil { 
  4.     errors.New("query incur error"
  5. for rows.Next(){ 
  6.     e := rows.Scan(user.sex, user.phone, user.nameuser.id,  
  7.       user.age) 
  8.     if e != nil{ 
  9.         fmt.Println(json.Marshal(user)) 
  10.     } 
  11. rows.Close() 
  12. // 單行查詢操作 
  13. DB.QueryRow("select * from user where id=1").Scan(user.age,  
  14.   user.id, user.nameuser.phone, user.sex) 

執(zhí)行流程 

  1. 使用db.Query()來(lái)發(fā)送查詢到數(shù)據(jù)庫(kù),獲取結(jié)果集Rows,并檢查錯(cuò)誤。
  2. 使用rows.Next()作為循環(huán)條件,迭代讀取結(jié)果集。
  3. 使用rows.Scan從結(jié)果集中獲取一行結(jié)果。
  4. 使用rows.Err()在退出迭代后檢查錯(cuò)誤。
  5. 使用rows.Close()關(guān)閉結(jié)果集,釋放連接。

增刪改和Exec

通常不會(huì)約束你查詢必須用Query,只是Query會(huì)返回結(jié)果集,而Exec不會(huì)返回。所以如果你執(zhí)行的是增刪改操作一般用Exec會(huì)好一些。Exec返回的結(jié)果是Result,Result接口允許獲取執(zhí)行結(jié)果的元數(shù)據(jù):

  1. type Result interface { 
  2.     // 用于返回自增ID,并不是所有的關(guān)系型數(shù)據(jù)庫(kù)都有這個(gè)功能。 
  3.     LastInsertId() (int64, error) 
  4.     // 返回受影響的行數(shù)。 
  5.     RowsAffected() (int64, error) 

 準(zhǔn)備查詢

如果你現(xiàn)在想使用占位符的功能,where 的條件想以參數(shù)的形式傳入,Go提供了db.Prepare語(yǔ)句來(lái)幫你綁定。準(zhǔn)備查詢的結(jié)果是一個(gè)準(zhǔn)備好的語(yǔ)句(prepared statement),語(yǔ)句中可以包含執(zhí)行時(shí)所需參數(shù)的占位符(即綁定值)。準(zhǔn)備查詢比拼字符串的方式好很多,它可以轉(zhuǎn)義參數(shù),避免SQL注入。同時(shí),準(zhǔn)備查詢對(duì)于一些數(shù)據(jù)庫(kù)也省去了解析和生成執(zhí)行計(jì)劃的開(kāi)銷,有利于性能。

占位符

PostgreSQL使用$N作為占位符,N是一個(gè)從1開(kāi)始遞增的整數(shù),代表參數(shù)的位置,方便參數(shù)的重復(fù)使用。MySQL使用?作為占位符,SQLite兩種占位符都可以,而Oracle則使用:param1的形式。

  1. MySQL               PostgreSQL            Oracle 
  2. =====               ==========            ====== 
  3. WHERE col = ?       WHERE col = $1        WHERE col = :col 
  4. VALUES(?, ?, ?)     VALUES($1, $2, $3)    VALUES(:val1, :val2, :val3) 
  5. stmt, e := DB.Prepare("select * from user where id=?"
  6. query, e := stmt.Query(1) 
  7. query.Scan() 

 事務(wù)的使用

通過(guò)db.Begin()來(lái)開(kāi)啟一個(gè)事務(wù),Begin方法會(huì)返回一個(gè)事務(wù)對(duì)象Tx。在結(jié)果變量Tx上調(diào)用Commit()或者Rollback()方法會(huì)提交或回滾變更,并關(guān)閉事務(wù)。在底層,Tx會(huì)從連接池中獲得一個(gè)連接并在事務(wù)過(guò)程中保持對(duì)它的獨(dú)占。事務(wù)對(duì)象Tx上的方法與數(shù)據(jù)庫(kù)對(duì)象sql.DB的方法一一對(duì)應(yīng),例如Query,Exec等。事務(wù)對(duì)象也可以準(zhǔn)備(prepare)查詢,由事務(wù)創(chuàng)建的準(zhǔn)備語(yǔ)句會(huì)顯式綁定到創(chuàng)建它的事務(wù)。

  1. //開(kāi)啟事務(wù) 
  2. tx, err := DB.Begin() 
  3. if err != nil { 
  4.     fmt.Println("tx fail"
  5. //準(zhǔn)備sql語(yǔ)句 
  6. stmt, err := tx.Prepare("DELETE FROM user WHERE id = ?"
  7. if err != nil { 
  8.     fmt.Println("Prepare fail"
  9.     return false 
  10. //設(shè)置參數(shù)以及執(zhí)行sql語(yǔ)句 
  11. res, err := stmt.Exec(user.id) 
  12. if err != nil { 
  13.     fmt.Println("Exec fail"
  14.     return false 
  15. //提交事務(wù) 
  16. tx.Commit() 
  17.  
  18. 我們來(lái)一個(gè)完整的sql操作:package main 
  19.  
  20. import ( 
  21.     "database/sql" 
  22.     "encoding/json" 
  23.     "fmt" 
  24.     _ "github.com/go-sql-driver/mysql" 
  25.     "github.com/pkg/errors" 
  26.     "strings" 
  27.  
  28. //數(shù)據(jù)庫(kù)配置 
  29. const ( 
  30.     userName = "root" 
  31.     password = "123456" 
  32.     ip       = "127.0.0.1" 
  33.     port     = "3306" 
  34.     dbName   = "test" 
  35.  
  36. //Db數(shù)據(jù)庫(kù)連接池 
  37. var DB *sql.DB 
  38.  
  39. type User struct { 
  40.     id    int64 
  41.     name  string 
  42.     age   int8 
  43.     sex   int8 
  44.     phone string 
  45.  
  46. //注意方法名大寫(xiě),就是public 
  47. func InitDB() { 
  48.     //構(gòu)建連接:"用戶名:密碼@tcp(IP:端口)/數(shù)據(jù)庫(kù)?charset=utf8" 
  49.     path := strings.Join([]string{userName, ":"password"@tcp(", ip, ":", port, ")/", dbName, "?charset=utf8"}, ""
  50.     //打開(kāi)數(shù)據(jù)庫(kù),前者是驅(qū)動(dòng)名,所以要導(dǎo)入:_ "github.com/go-sql-driver/mysql" 
  51.     DB, _ = sql.Open("mysql", path) 
  52.     //設(shè)置數(shù)據(jù)庫(kù)最大連接數(shù) 
  53.     DB.SetConnMaxLifetime(100) 
  54.     //設(shè)置上數(shù)據(jù)庫(kù)最大閑置連接數(shù) 
  55.     DB.SetMaxIdleConns(10) 
  56.     //驗(yàn)證連接 
  57.     if err := DB.Ping(); err != nil { 
  58.         fmt.Println("open database fail"
  59.         return 
  60.     } 
  61.     fmt.Println("connnect success"
  62.  
  63. //查詢操作 
  64. func Query() { 
  65.     var user User 
  66.     rows, e := DB.Query("select * from user where id in (1,2,3)"
  67.     if e == nil { 
  68.         errors.New("query incur error"
  69.     } 
  70.     for rows.Next() { 
  71.         e := rows.Scan(user.sex, user.phone, user.nameuser.id, user.age) 
  72.         if e != nil { 
  73.             fmt.Println(json.Marshal(user)) 
  74.         } 
  75.     } 
  76.     rows.Close() 
  77.     DB.QueryRow("select * from user where id=1").Scan(user.age, user.id, user.nameuser.phone, user.sex) 
  78.  
  79.     stmt, e := DB.Prepare("select * from user where id=?"
  80.     query, e := stmt.Query(1) 
  81.     query.Scan() 
  82.  
  83. func DeleteUser(user User) bool { 
  84.     //開(kāi)啟事務(wù) 
  85.     tx, err := DB.Begin() 
  86.     if err != nil { 
  87.         fmt.Println("tx fail"
  88.     } 
  89.     //準(zhǔn)備sql語(yǔ)句 
  90.     stmt, err := tx.Prepare("DELETE FROM user WHERE id = ?"
  91.     if err != nil { 
  92.         fmt.Println("Prepare fail"
  93.         return false 
  94.     } 
  95.     //設(shè)置參數(shù)以及執(zhí)行sql語(yǔ)句 
  96.     res, err := stmt.Exec(user.id) 
  97.     if err != nil { 
  98.         fmt.Println("Exec fail"
  99.         return false 
  100.     } 
  101.     //提交事務(wù) 
  102.     tx.Commit() 
  103.     //獲得上一個(gè)insert的id 
  104.     fmt.Println(res.LastInsertId()) 
  105.     return true 
  106.  
  107. func InsertUser(user User) bool { 
  108.     //開(kāi)啟事務(wù) 
  109.     tx, err := DB.Begin() 
  110.     if err != nil { 
  111.         fmt.Println("tx fail"
  112.         return false 
  113.     } 
  114.     //準(zhǔn)備sql語(yǔ)句 
  115.     stmt, err := tx.Prepare("INSERT INTO user (`name`, `phone`) VALUES (?, ?)"
  116.     if err != nil { 
  117.         fmt.Println("Prepare fail"
  118.         return false 
  119.     } 
  120.     //將參數(shù)傳遞到sql語(yǔ)句中并且執(zhí)行 
  121.     res, err := stmt.Exec(user.nameuser.phone) 
  122.     if err != nil { 
  123.         fmt.Println("Exec fail"
  124.         return false 
  125.     } 
  126.     //將事務(wù)提交 
  127.     tx.Commit() 
  128.     //獲得上一個(gè)插入自增的id 
  129.     fmt.Println(res.LastInsertId()) 
  130.     return true 
  131.  
  132. func main() { 
  133.     InitDB() 
  134.     Query() 
  135.     defer DB.Close() 

參考

https://www.cnblogs.com/rickiyang/p/11074180.html

 

責(zé)任編輯:姜華 來(lái)源: JavaEdge
相關(guān)推薦

2010-05-14 11:12:16

連接MySql

2021-08-02 10:53:28

PythonMySQL數(shù)據(jù)庫(kù)

2009-11-24 16:48:15

PHP mysqli

2017-09-11 19:30:44

MySQLCmd命令連接數(shù)據(jù)庫(kù)

2018-02-26 20:00:00

編程語(yǔ)言JavaMySQL

2009-06-01 09:57:43

netbeans連接數(shù)netbeans數(shù)據(jù)庫(kù)netbeans連接m

2009-07-07 14:56:33

JSP連接MySQL

2020-09-22 15:56:31

Java

2011-05-26 13:42:50

MFC連接MySql數(shù)據(jù)庫(kù)

2011-07-27 13:58:48

EclipseMySQL

2024-08-19 11:45:18

2017-11-27 11:41:06

python數(shù)據(jù)庫(kù)數(shù)據(jù)分析

2024-01-02 08:47:42

2023-08-29 07:35:15

2010-06-07 15:24:34

Java連接MYSQL

2024-12-04 16:12:31

2011-07-05 10:16:16

Qt 數(shù)據(jù)庫(kù) SQLite

2023-12-05 07:26:21

Golang項(xiàng)目結(jié)構(gòu)

2024-05-28 00:00:30

Golang數(shù)據(jù)庫(kù)

2011-07-18 14:00:29

RailsOracle
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)

主站蜘蛛池模板: 国产日韩一区二区 | 欧美二区三区 | 伊人导航 | 日韩精品中文字幕一区二区三区 | 麻豆a级片 | 久久精品国产免费高清 | 国产视频一二三区 | 亚洲手机在线 | 国产精品亚洲成在人线 | 七七婷婷婷婷精品国产 | 久久黄色 | 国产不卡在线播放 | 九色在线观看 | 在线视频a| 香蕉视频1024 | 最近日韩中文字幕 | 成人一区二区三区在线 | 岛国午夜 | 91在线看 | 美女黄18岁以下禁止观看 | 精品国产一区二区三区久久久久久 | 国产视频黄色 | 中文字幕一区二区三 | 黄色三级毛片 | 欧美视频一区 | 国产一区二区电影网 | 成人av电影网 | 国产成人jvid在线播放 | 午夜日韩 | 成人国产精品久久久 | 成人一区二 | 日本aa毛片a级毛片免费观看 | 国产精品久久久亚洲 | 国产1区2区3区 | 欧美一区二区三区免费电影 | 一区二区免费 | 伊人最新网址 | 欧美中文字幕在线观看 | 久久久国产一区二区三区 | 色眯眯视频在线观看 | 天天操,夜夜爽 |