golang mysql 单例,如何在GoLang中创建单例数据库类

EDITED Solved:How should i create singleton DBManager class in GoLang.

I Referred few code sample of how to create go singleton but i wish to have methods in those and call them on their singleton reference. My Code is as follows

package dbprovider

import (

"github.com/jinzhu/gorm"

_"github.com/jinzhu/gorm/dialects/sqlite"

"rest/article"

"log"

)

type DBOperations interface {

AddArticle(article *article.Article)

}

type DBManager struct {

db *gorm.DB

isInitialized bool

}

var dbManagerInstance = new()

func GetDBManager() DBManager {

return dbManagerInstance

}

func new() DBManager {

localDbRef, err := gorm.Open("sqlite3", "../articles.db")

if (err != nil) {

panic("Error initializing db")

} else {

log.Print("DB Initialized successfully")

}

return DBManager{db:localDbRef, isInitialized:true}

}

func (dbManager DBManager) AddArticle(article article.Article) (err error) {

if (dbManager.isInitialized) {

tx := dbManager.db.Begin()

//dbManager.db.NewRecord(article)

//dbManager.db.Commit()

tx.NewRecord(article)

tx.Commit()

errs := dbManager.db.GetErrors()

if (len(errs) > 0) {

err = errs[0]

} else {

log.Print("No error in this transactions")

}

}

return

}

With new answer i have updated this question including answer. But i have few queries. How to cathc and return exception from gorm.Create(..)

解决方案

One way is to create an exported interface with the methods, and make the implementing type unexported. Create a global variable of the interface type, and initialize it with a package init() function. You don't need any synchronization as the package init() function will run only once, safely.

Package init() functions are executed once, automatically, by the runtime, before you could refer to anything from the package. For details, see Spec: Package initialization.

For example:

package dbprovider

type Manager interface {

AddArticle(article *article.Article) error

// Add other methods

}

type manager struct {

db *gorm.DB

}

var Mgr Manager

func init() {

db, err := gorm.Open("sqlite3", "../articles.db")

if err != nil {

log.Fatal("Failed to init db:", err)

}

Mgr = &manager{db: db}

}

func (mgr *manager) AddArticle(article *article.Article) (err error) {

mgr.db.Create(article)

if errs := mgr.db.GetErrors(); len(errs) > 0 {

err = errs[0]

}

return

}

Using it:

import "dbprovider"

if err := dbprovider.Mgr.AddArticle(someArticle); err != nil {

// Handle error

}

You could also do it without an init() function, e.g.:

var Mgr = newManager()

func newManager() Manager {

db, err := gorm.Open("sqlite3", "../articles.db")

if err != nil {

log.Fatal("Failed to init db:", err)

}

return &manager{db: db}

}

With this you may decide to make newManager() exported and users of your package could decide to use the shared Mgr instance, or they could create another Manager, e.g. for testing purposes.

Notes: Mgr is an exported global variable, and it is possible to assign a new value to it by other packages (e.g. dbprovider.Mgr = nil). If you want to avoid this, you have to make it unexported, and provide a "getter" function for it, e.g.:

var mgr = newManager()

func Mgr() Manager { return mgr }

And using it:

err := dbprovider.Mgr().AddArticle(someArticle)

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值