以前,已经讲过不少Golang的东西,好比基础语法,mysql的使用,redis的使用等等,感兴趣的能够看看之前的文章,https://www.cnblogs.com/zhangweizhong/category/1275863.html,html
今天就用从头写一个完整的go的示例项目吧。mysql
下图这种架构模式相信你们应该十分清楚git
Controller"基类"封装github
package framework
type Controller struct {
Data interface{}
}
UserController定义了用户注册,登陆和查询等简单的三个接口web
package controller
import (
"net/http"
"micro-cloud/service"
"micro-cloud/utils"
"micro-cloud/framework"
)
/**
* r.PostFormValue : 能够解析 Post/PUT Content-Type=application/x-www-form-urlencoded 或 Content-Type=multipart/form-data
*/
type UserConterller struct {
}
var userService = new(service.UserService)
func (p *UserConterller) Router(router *framework.RouterHandler) {
router.Router("/register", p.register)
router.Router("/login", p.login)
router.Router("/findAll", p.findAll)
}
//POST Content-Type=application/x-www-form-urlencoded
func (p *UserConterller) register(w http.ResponseWriter, r *http.Request) {
username := r.PostFormValue("username")
password := r.PostFormValue("password")
if utils.Empty(username) || utils.Empty(password) {
microcloud.ResultFail(w, "username or password can not be empty")
return
}
id := userService.Insert(username, password)
if id <= 0 {
microcloud.ResultFail(w, "register fail")
return
}
microcloud.ResultOk(w, "register success")
}
//POST Content-Type=application/x-www-form-urlencoded
func (p *UserConterller) login(w http.ResponseWriter, r *http.Request) {
username := r.PostFormValue("username")
password := r.PostFormValue("password")
if utils.Empty(username) || utils.Empty(password) {
microcloud.ResultFail(w, "username or password can not be empty")
return
}
users := userService.SelectUserByName(username)
if len(users) == 0 {
microcloud.ResultFail(w, "user does not exist")
return
}
if users[0].Password != password {
microcloud.ResultFail(w, "password error")
return
}
microcloud.ResultOk(w, "login success")
}
// GET/POST
func (p *UserConterller) findAll(w http.ResponseWriter, r *http.Request) {
users := userService.SelectAllUser()
framework.ResultJsonOk(w, users)
}
package dao import ( "micro-cloud/model" "fmt" ) type UserDao struct { } func (p *UserDao) Insert(user *model.User) int64 { result, err := framework.DB.Exec("INSERT INTO user(`username`,`password`,`create_time`) value(?,?,?)", user.Username, user.Password, user.CreateTime) if err != nil { fmt.Println("Insert error") return 0 } id, err := result.LastInsertId() if err != nil { fmt.Println("Insert error") return 0 } return id } func (p *UserDao) SelectUserByName(username string) []model.User { rows, err := framework.DB.Query("SELECT * FROM user WHERE username = ?", username) if err != nil { fmt.Println("selectuserbyname error") return nil } var users []model.User for rows.Next() { var user model.User err := rows.Scan(&user.ID, &user.Username, &user.Password, &user.CreateTime) if err != nil { fmt.Println("selectuserbyname error") continue } users = append(users, user) } rows.Close() return users } func (p *UserDao) SelectAllUser() []model.User { rows, err := framework.DB.Query("SELECT * FROM user") if err != nil { fmt.Println("SelectAllUser error") return nil } var users []model.User for rows.Next() { var user model.User err := rows.Scan(&user.ID, &user.Username, &user.Password, &user.CreateTime) if err != nil { fmt.Println("SelectAllUser error") continue } users = append(users, user) } rows.Close() return users }
package model import "time" type User struct { ID uint `json:"id"` Username string `json:"username"` Password string `json:"-"` CreateTime time.Time `json:"create_time"` }
数据库操做类的实现redis
package microcloud
import (
"database/sql"
"log"
"strings" _ "github.com/go-sql-driver/mysql" ) //数据库的配置 const ( username = "admin" password = "123456" ip = "10.2.1.5" port = "3306" dbName = "microcloud" driverName = "mysql" ) //DB数据库链接池 var DB *sql.DB func InitDB() { //构建链接:"用户名:密码@tcp(IP:端口)/数据库?charset=uft8" //注意:要想解析time.Time类型,必需要设置parseTime=True path := strings.Join([]string{username, ":", password, "@tcp(", ip, ":", port, ")/", dbName, "?charset=utf8&parseTime=True&loc=Local"}, "") //打开数据库,前者是驱动名,因此要导入:_"github.com/go-sql-driver/mysql" DB, _ = sql.Open(driverName, path) //设置数据库最大链接数 DB.SetConnMaxLifetime(100) //设置数据库最大闲置链接数 DB.SetMaxIdleConns(10) //验证链接 if err := DB.Ping(); err != nil { log.Panic(err) } log.Println("database connect success") } func CreateTable() { userTable := "CREATE TABLE IF NOT EXISTS `user`(" + "`id` INT UNSIGNED AUTO_INCREMENT," + "`username` VARCHAR(20) NOT NULL," + "`password` VARCHAR(40) NOT NULL," + "`create_time` DATETIME," + "PRIMARY KEY ( `id` )" + ")ENGINE=InnoDB DEFAULT CHARSET=utf8;" _, err := DB.Exec(userTable) if err != nil { log.Panic(err) } }
server := &http.Server{ Addr: ":8215", Handler: microcloud.Router, ReadTimeout: 5 * time.Second, } RegiterRouter(microcloud.Router) err := server.ListenAndServe() if err != nil { log.Panic(err) }
路由处理的实现,其实也就是一个转发的功能数据库
type RouterHandler struct { } var mux = make(map[string]func(http.ResponseWriter,*http.Request)) func (p *RouterHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) { fmt.Println(r.URL.Path) if fun, ok := mux[r.URL.Path]; ok { fun(w, r) return } //静态资源 if strings.HasPrefix(r.URL.Path,constant.STATIC_BAES_PATH){ if fun, ok := mux[constant.STATIC_BAES_PATH]; ok { fun(w, r) return } } http.Error(w, "error URL:"+r.URL.String(), http.StatusBadRequest) } func (p *RouterHandler) Router(relativePath string, handler func(http.ResponseWriter, *http.Request)) { mux[relativePath] = handler }
执行 go run main.go 以后,打开Postman,调相关的接口json
如下就是访问API的请求与响应api
/findAll 接口
/register 接口
以上,用Go语言实现webapi 的例子,已经介绍完了,虽然比较简单,session,权限验证等都没有加。可是最主要的功能已经讲完了,感兴趣的能够从头编写下相关的代码。
完整例子下载:microcloud.rar