做网站宣传有用吗,网店运营推广具体内容,网页设计兼职,wordpress多站共享授权码config #xff1a;敏感的配置一般都是在配置中心配置#xff0c;比如consul或者阿波罗上面controller #xff1a;写一些handler的#xff0c;拿到参数要去调用service层的逻辑。#xff08;只负责接受参数#xff0c;怎么绑定参数#xff0c;要去调用哪个service的敏感的配置一般都是在配置中心配置比如consul或者阿波罗上面controller 写一些handler的拿到参数要去调用service层的逻辑。只负责接受参数怎么绑定参数要去调用哪个service的handler的一个入口serviceservice层才是正真的业务处理层。调用dao层的增删改查操作dao只关心数据库的一个操作其实你有数据库和es都在dao层db负责数据库中间件层的初始化middleware存放gin的中间件在注册路由的时候将中间件引入进来 go常用项目结构 dao层只关心数据库的操作db层只做Mysql层的初始化。model去定义表结构定义中间表多对多的结构体。
中间件有个token的验证。
controller层先去找路由这个项目有多少的接口router.go每个接口到哪些对应的handler。
controller层接收完参数之后再看调用到哪个serviceservice再去干了什么增删改查操作。
数据库增删改查就去dao层查看。
这一层一层非常的分明。上面可以理解为日常开发的目录结构最佳实践。 创建数据库 mysql create database books charset utf8; 图书管理服务 用户服务登录注册 书籍服务对书籍的增删改查的操作 主要有两个维度一个是书籍维度一个是用户维度。用户维度要去写两个功能一个功能是登入一个功能是注册。 注册就是数据库插入一条数据。登入就是一个校验。配合token中间件去做一个校验。 在新增书籍的时候要去关联用户这本书属于谁书籍的增删改查都需要关联这个用户。这里就涉及到1对多的关系。 开发的时候是从最底层开始往上开发。第一个先去定义数据库的model层。 model层 定义数据库结构 binding标签表示是必填项token是可以为空的因为一开始注册的时候token的为空。只有登入的时候才有token。
user.go
package modeltype User struct {ID int64 gorm:primaryKey json:idUserName string gorm:not null json:username binding:requiredPassWord string gorm:not null json:password binding:requiredToken string json:token
}func (*User) TableName() string {return user
}
book.go
package modeltype Book struct {ID int64 gorm:primaryKey json:idName string gorm:not null json:name binding:requiredDesc string json:descUsers []User gorm:many2many:book_users
}func (*Book) TableName() string {return book
}
多对多关系可以在user这层定义也行在book这一层定义也可以。这个主要看你的一个实际使用的场景这里在book模型里面去定义就行了。
还需要去定义一个中间表的模型user_m2m_book.go
package modeltype BookUser struct {UserID int64 gorm:primaryKeyBookID int64 gorm:primaryKey
}这里不需要自定义表名它只有一个主键也没有其他属性了。它也是永了外键也是使用了那两个模型的主键。 DB层 模型定义好之后去做数据库的初始化这里需要预留因为可能不仅仅只有MySQL的初始化。 这样赋值到一个全局变量之后使用mysql.DB就可以在任何地方去使用了。
package mysqlimport (book/modelfmtgorm.io/driver/mysqlgorm.io/gorm
)// DB 要将DB实例放到全局变量,这样就可以使用mysql.DB在任何地方去使用了.
var DB *gorm.DBfunc InitMysql() {dsn : root:7PXjAkY!nlRtcp(192.168.11.128:3306)/books?charsetutf8mb4parseTimeTruelocLocaldb, err : gorm.Open(mysql.Open(dsn), gorm.Config{})//这个地方Log和panic都是可以的if err ! nil {fmt.Println(err)}DB dberr DB.AutoMigrate(model.Book{}, model.User{}, model.BookUser{})if err ! nil {fmt.Println(err)}
} DAO层 初始化好之后在dao层开始写dao层的操作就是数据库的增删改查 package daoimport (book/db/mysqlbook/modelerrorsfmtgithub.com/wonderivan/loggergorm.io/gorm
)//定义user结构体以及User变量能够直接跨包调用user下面的方法
//只需要一次初始化即可,不用每次调用的都是先初始化var User usertype user struct {
}// Add 新增 用于注册
func (*user) Add(user *model.User) error {if tx : mysql.DB.Create(user); tx.Error ! nil {//打印错误提示logger.Error(fmt.Sprintf(添加User失败,错误信息:%s, tx.Error))return errors.New(fmt.Sprintf(添加User失败,错误信息:%s, tx.Error))}return nil
}// Has 查询 基于name 用于新增
func (*user) Has(name string) (*model.User, bool, error) {//初始化要申请内存不然会报错data : model.User{}tx : mysql.DB.Where(username ?, name).Find(data)//如果记录没有查询到报错是从tx.error里面拿到的相当于tx.Error gorm.ErrRecordNotFoundif errors.Is(tx.Error, gorm.ErrRecordNotFound) {return nil, false, nil}//等会去调用的时候不会先去判断有没有要先去判断error有error就是真正的错误没有判断是不是falseif tx.Error ! nil {logger.Error(fmt.Sprintf(查询用户失败,错误信息:%s, tx.Error))return nil, false, errors.New(fmt.Sprintf(查询用户失败,错误信息:%s, tx.Error))}return data, true, nil
}// GetByToken 基于token查询用于中间件token校验
func (*user) GetByToken(token string) (*model.User, bool, error) {//初始化要申请内存不然会报错data : model.User{}tx : mysql.DB.Where(token ?, token).Find(data)//如果记录没有查询到报错是从tx.error里面拿到的相当于tx.Error gorm.ErrRecordNotFoundif errors.Is(tx.Error, gorm.ErrRecordNotFound) {return nil, false, nil}if tx.Error ! nil {logger.Error(fmt.Sprintf(查询用户失败%s, tx.Error))return nil, false, errors.New(fmt.Sprintf(查询用户失败%v/n, tx.Error))}return data, true, nil
}// UPDateToken 更新token,这里其实就是找到user的条件去更新就行了
// 第一个参数可以是id也可以是用户名如果用户名唯一只要保证找到指定用户即可
// 除了查询的操作以外增删改只需要返回一个error即可判断操作有没有成功
func (*user) UPDateToken(user *model.User, token string) error {tx : mysql.DB.Model(user).Where(username ? and password ?, user.UserName, user.PassWord).Update(token, token)if tx.Error ! nil {logger.Error(fmt.Sprintf(更新user token失败,错误信息:%s, tx.Error))return errors.New(fmt.Sprintf(更新user token失败,错误信息:%s, tx.Error))}return nil
} service层 和dao层一样都定义了相同的结构体不同的包使用相同的结构体变量但是点出来的方法都是不同的。