这篇文章我们来重构一下之前写的Todolist项目,包括项目结构,代码逻辑
项目地址:https://github.com/CocaineCong/TodoList
TodoList/
├── api
├── cache
├── conf
├── middleware
├── model
├── pkg
│ ├── e
│ └── util
├── routes
├── serializer
└── service
这个项目结构看起来没啥问题,但实际使用的过程中问题很大!
所以我们这里应该用一个文件统一放置这些东西
。req
,resp
等待所需要的结构体。注意一点:types,consts包都是只能单方面引入,也就是这两个包只能被引用,不能引用其他包的代码,否则容易变成循环导包的情况
TodoList/
├── api
├── cmd
├── conf
├── consts
├── docs
├── middleware
├── pkg
│ ├── e
│ └── util
├── routes
├── repository
│ ├── cache
│ └── db
│ ├── dao
│ └── model
├── routes
├── service
└── types
这部分我们以Task模块举例子
先看代码:
func CreateTask(c *gin.Context) {
createService := service.CreateTaskService{}
chaim, _ := util.ParseToken(c.GetHeader("Authorization"))
if err := c.ShouldBind(&createService); err == nil {
res := createService.Create(chaim.Id)
c.JSON(200, res)
} else {
c.JSON(400, ErrorResponse(err))
util.LogrusObj.Info(err)
}
}
这个代码咋一看,就很多问题。
放在中间件中统一解析,然后存到context中,进行上下游的流动存储
。链路没法跟踪了
。返回error值
,这是go语言的特色( 也不知道go为啥这种设计,一堆if err != nil )http包中的常量
。修改之后的controller层代码如下所示:
func CreateTaskHandler() gin.HandlerFunc {
return func(ctx *gin.Context) {
var req types.CreateTaskReq
if err := ctx.ShouldBind(&req); err == nil {
// 参数校验
l := service.GetTaskSrv()
resp, err := l.CreateTask(ctx.Request.Context(), &req)
if err != nil {
ctx.JSON(http.StatusInternalServerError, ErrorResponse(err))
return
}
ctx.JSON(http.StatusOK, resp)
} else {
util.LogrusObj.Infoln(err)
ctx.JSON(http.StatusBadRequest, ErrorResponse(err))
}
}
}
context进行上下游传递
// 创建任务的服务
type CreateTaskService struct {
Title string `form:"title" json:"title" binding:"required,min=2,max=100"`
Content string `form:"content" json:"content" binding:"max=1000"`
Status int `form:"status" json:"status"` // 0 待办 1已完成
}
func (service *CreateTaskService) Create(id uint) serializer.Response {
var user model.User
model.DB.First(&user, id)
task := model.Task{
User: user,
Uid: user.ID,
Title: service.Title,
Content: service.Content,
Status: 0,
StartTime: time.Now().Unix(),
}
code := e.SUCCESS
err := model.DB.Create(&task).Error
if err != nil {
util.LogrusObj.Info(err)
code = e.ErrorDatabase
return serializer.Response{
Status: code,
Msg: e.GetMsg(code),
Error: err.Error(),
}
}
return serializer.Response{
Status: code,
Data: serializer.BuildTask(task),
Msg: e.GetMsg(code),
}
}
那么我们应该怎么做改呢?
首先我们应该改造我们的对象模式,使用sync.Once来进行对象的创建,原本是饿汉式的单例模式,现在是懒汉式的单例模式
。
懒汉式指的是在第一次访问单例对象时才进行实例化,而不是在程序启动时就进行实例化。
在下面的这个示例中,当第一次调用GetTaskSrv函数时,才会执行once.Do方法并初始化单例对象,从而实现了懒加载的效果
。
var TaskSrvIns *TaskSrv
var TaskSrvOnce sync.Once
type TaskSrv struct {
}
func GetTaskSrv() *TaskSrv {
TaskSrvOnce.Do(func() {
TaskSrvIns = &TaskSrv{}
})
return TaskSrvIns
}
在上面的代码中,使用了一个名为TaskSrvIns的全局变量来存储单例实例,once变量则用于确保GetTaskSrv函数只会被执行一次。
GetTaskSrv函数中,调用了once.Do方法,并传入一个匿名函数,用于初始化TaskSrvIns变量。这样,在第一次调用GetTaskSrv函数时,匿名函数会被执行,创建一个新的TaskSrv实例,并将其赋值给TaskSrvIns变量。之后,再次调用GetTaskSrv函数时,直接返回已经创建的TaskSrvIns变量,从而保证整个应用程序中只存在一个TaskSrv实例。
使用该实现方式可以有效地避免因为多线程操作而导致的线程安全问题,同时又能保证单例对象只会被创建一次
。
那么我们的函数就变成了这样:
传入context,定义的req结构体,返回了resp和error
func (s *TaskSrv) CreateTask(ctx context.Context, req *types.CreateTaskReq) (resp interface{}, err error) {}
之后我们抽离出dao层专门处理db的操作。
user, err := dao.NewUserDao(ctx).FindUserByUserId(u.Id)
task := &model.Task{
User: *user,
Uid: user.ID,
Title: req.Title,
Content: req.Content,
Status: 0,
StartTime: time.Now().Unix(),
}
err = dao.NewTaskDao(ctx).CreateTask(task)
然后再抽离出返回值
return ctl.RespSuccess(), nil
抽离获取用户的部分,我们后面再说。
完整代码:
var TaskSrvIns *TaskSrv
var TaskSrvOnce sync.Once
type TaskSrv struct {
}
func GetTaskSrv() *TaskSrv {
TaskSrvOnce.Do(func() {
TaskSrvIns = &TaskSrv{}
})
return TaskSrvIns
}
func (s *TaskSrv) CreateTask(ctx context.Context, req *types.CreateTaskReq) (resp interface{}, err error) {
u, err := ctl.GetUserInfo(ctx)
if err != nil {
util.LogrusObj.Info(err)
return
}
user, err := dao.NewUserDao(ctx).FindUserByUserId(u.Id)
if err != nil {
util.LogrusObj.Info(err)
return
}
task := &model.Task{
User: *user,
Uid: user.ID,
Title: req.Title,
Content: req.Content,
Status: 0,
StartTime: time.Now().Unix(),
}
err = dao.NewTaskDao(ctx).CreateTask(task)
if err != nil {
util.LogrusObj.Info(err)
return
}
return ctl.RespSuccess(), nil
}
我们先定义一下所需要的用户信息,这里我们只需要用户的id,所以定义这个id就可以了,后续如果需要用户名字或是其他信息,都是可以加上去的。
type UserInfo struct {
Id uint `json:"id"`
}
新建一个context,以便后续插入到context中,进行上下游的传递。
type key int
var userKey key
func NewContext(ctx context.Context, u *UserInfo) context.Context {
return context.WithValue(ctx, userKey, u)
}
从context中获取信息,进行返回
func FromContext(ctx context.Context) (*UserInfo, bool) {
u, ok := ctx.Value(userKey).(*UserInfo)
return u, ok
}
func GetUserInfo(ctx context.Context) (*UserInfo, error) {
user, ok := FromContext(ctx)
if !ok {
return nil, errors.New("获取用户信息错误")
}
return user, nil
}
所以我们在middleware层中可以把我们的new context并携带我们解析token的信息,加入到context流中进行上下游的流动。
c.Request = c.Request.WithContext(ctl.NewContext(c.Request.Context(), &ctl.UserInfo{Id: claims.Id}))
完整代码:
type key int
var userKey key
type UserInfo struct {
Id uint `json:"id"`
}
func GetUserInfo(ctx context.Context) (*UserInfo, error) {
user, ok := FromContext(ctx)
if !ok {
return nil, errors.New("获取用户信息错误")
}
return user, nil
}
func NewContext(ctx context.Context, u *UserInfo) context.Context {
return context.WithValue(ctx, userKey, u)
}
func FromContext(ctx context.Context) (*UserInfo, bool) {
u, ok := ctx.Value(userKey).(*UserInfo)
return u, ok
}