likes
comments
collection
share

02-用Go语言标准库实现Web服务之创建路由

作者站长头像
站长
· 阅读数 45

在这小节,我遇到了困难,我不知道该如何用那种方式来解说项目,如果直接直接粘贴代码,一下子全是代码脑瓜子嗡嗡的。拆分来讲,又怕串联不起来,我该以哪一种方式来述说,思考良久最终确定先提供代码仓库,再分解来讲。各位观众老爷们,如果你们有更好的方式请提供给我参考参考,非常感谢😎~

代码仓库在这👇👇👇重要的事情说三遍 EBook 代码仓库 记得给个Star😘😘😘 代码仓库在这👇👇👇重要的事情说三遍 EBook 代码仓库 记得给个Star😍😍😍

话不多说,回归正题,在上一节中创建了项目;现在打开internal/app/app.go中创建一个application结构体来管理api服务:

// application 一个管理API服务的结构体
type application struct {
    cfg   config.AppConfig
    store dbrepo.Repository
}

在上面其中 dbrepo.Repository 数据库操作层,先创建一个结构体,后续实现

  • internal/dbrepo/repository.go
type Repository struct {
}
func NewRepository(db *sql.DB) Repository {
    return Repository{
        Book: NewBookRepo(db),
    }
}

在这里我希望application实例管理整个api服务,包括启动服务、关闭服务、路由注册、路由处理、中间件等功能。

因此有一个newApplication函数,做初始化服务工作,首先要初始化就是加载配置文件,在系统中配置文件是一个关键信息,要最先初始化,从而保证后面用到配置的对象能够正常创建。

func newApplication() *application {
	// 解析命令行参数
	var cfgPath string
	flag.StringVar(&cfgPath, "path", "configs/app.json", "config 配置文件路径")
	flag.Parse()

	// 加载app配置
	cfg, err := config.LoadAppConfig(cfgPath)
	if err != nil {
		log.Fatal(err)
	}
        // 格式化输出一下配置,仅开发模式下
	cfg.Println()

	// 设置全局日志
	logger.SetGlobalLogger()

	// 连接mysql
	db, err := sql.Open("mysql", cfg.DSN)
	if err != nil {
		log.Fatal(err)
	}

	// ping 确定能连接上数据库
	if err = db.Ping(); err != nil {
		log.Fatal(err)
	}

	// 设置最大链接数、最大空闲数和最大空闲时间
	db.SetMaxOpenConns(cfg.MaxOpenConns)
	db.SetMaxIdleConns(cfg.MaxIdleConns)
	db.SetConnMaxIdleTime(cfg.MaxIdleTimeToDuration())

	a := &application{
		cfg:   cfg,
		store: dbrepo.NewRepository(db),
	}

	return a
}

上面函数一开始使用flag库来解析命令行参数,函数解析:flag.StringVar(解析后赋值给谁,参数名字,参数默认值,参数提示语),最后一定要用flag.Parse()解析参数。那么这个命令参数是怎么用的呢?简单,在启动的时候如下:

go run main.go -path=configs/app.jso

命令行参数使用介绍完毕,在这里顺便另一个场景,例如在别人代码中经常看到这样一个用法,如:

env := os.Getenv("APP_ENV")

那么这个APP_ENV环境变量在哪里配置的呢?这里介绍两种情况设置:

- 1. 在代码中设置 os.Setenv(key string, value string)
- 2. 启动程序通过命令设置如:
    APP_ENV=dev go run main.go

另外在这里我修改了maxOpenConns、maxIdleConns、maxIdleTime 配置,目前来讲对一个中小型项目来说刚运营时,这个配置是最合适的。后续可以根据服务流量调整。

{
    ...
    "maxOpenConns": 25,
    "maxIdleConns": 25,
    "maxIdleTime": "5m"
}

接下来就是创建一个http.Server 实例,使用这个实例来启动API服务。

func (a *application) serve() error {
    var address = fmt.Sprintf("0.0.0.0:%d", a.cfg.Port)

    // 创建 http.Server 用来启动 Web Server
    srv := http.Server{
            Addr:         address,
            Handler:      a.routes(),
            IdleTimeout:  time.Minute,
            ReadTimeout:  15 * time.Second,
            WriteTimeout: 30 * time.Second,
    }

    logger.InfoLog.Println("Starting API Server on: " + address)
    return srv.ListenAndServe()
}

func Serve() error {
    return newApplication().serve()
}

首先在这里func (a *application) serve() 用于内部创建一个 http.Server 和启动server; 而 func Serve() 是提供外部的main.go 中main函数调用。 很多开源项目都是这个做法,对导出去函数简化。

然后使用http.NewServeMux()创建路由:

  • internal/app/routes.go
package app

import "net/http"

func (a *application) routes() http.Handler {
	mux := http.NewServeMux()

	mux.HandleFunc("/v1/book", a.tmp)       // GET    获取一个图书 book?id=1
	mux.HandleFunc("/v1/book/post", a.tmp)  // POST   创建一个图书
	mux.HandleFunc("/v1/book/put", a.tmp)   // PUT    更新一个图书
	mux.HandleFunc("/v1/book/del", a.tmp)   // DELETE 删除一个图书
	mux.HandleFunc("/v1/book/list", a.tmp) // GET    获取图书列表 list?page_idx=1&pageNum=10

	return mux
}

func (a *application) tmp(w http.ResponseWriter, r *http.Request) {
	fmt.Fprintf(w, "%s:%s", r.Method, r.RequestURI)
}

最后就是启动程序~

  • main.go
package main

import (
    "log"
    "github.com/lightsaid/ebook/internal/app"
)

func main() {
    if err := app.Serve(); err != nil {
            log.Fatal(err)
    }
}

人生苦短 Let's Go

02-用Go语言标准库实现Web服务之创建路由 快去试试吧~

转载自:https://juejin.cn/post/7235237072440164407
评论
请登录