手把手教你Go+Vue前后端分离设计实践

举报
海风极客 发表于 2023/05/20 14:07:50 2023/05/20
【摘要】 在之前我曾写过一篇文章《手把手教你搭建Spring Boot+Vue前后端分离》,讲述了如何使用当下流行的Java后端框架Spring Boot和前端框架Vue来进行前后端分离设计,以及什么是前后端分离、跨越问题和设计流程等等,当时还是一名妥妥的Javaer,可是时过境迁,现在的我已然是一名十分活跃的Gopher,成为Gopher一段时间之后再回头看Java的代码,有三个问题甚是不解:怎么会...

在之前我曾写过一篇文章《手把手教你搭建Spring Boot+Vue前后端分离》,讲述了如何使用当下流行的Java后端框架Spring Boot和前端框架Vue来进行前后端分离设计,以及什么是前后端分离、跨越问题和设计流程等等,当时还是一名妥妥的Javaer,可是时过境迁,现在的我已然是一名十分活跃的Gopher,成为Gopher一段时间之后再回头看Java的代码,有三个问题甚是不解:

  • 怎么会有try…catch…这种东西?
  • 为什么每一行末尾一定要加分号…
  • 我的指针呢!?

哈哈开个玩笑,下面我们回归正题。

本次这篇文章可以称为是上一篇文章的姊妹篇,使用更加简洁的Go语言和它的Gin框架来设计后端,进行一次Go+Vue前后端分离实践,希望能够对Go语言的初学者起到良好的参考作用。

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gscZi00q-1682857177423)(一文搞懂Go+Vue前后端分离设计实践.assets/image-20230430160218534.png)]

这个是更新后的架构图,当然网络上可能会有更加详细的,大家可以先进行大致的了解。

1 为什么要进行前后端分离设计

答:一切都是为了解耦合!

前后端分离设计是一种软件开发方式,它将前端和后端设计为两个独立的模块,分别由不同的团队进行开发和维护。这种设计模式可以带来一些好处,包括:

  1. 提高代码质量:前后端分离可以使开发人员专注于实现业务逻辑,从而提高代码质量和可维护性。
  2. 减少耦合性:前后端分离可以减少代码之间的耦合性,从而减少后期维护的难度。
  3. 更好的测试和部署:前后端分离可以使得测试和部署更加容易,因为前后端代码是独立的。
  4. 提高性能:前后端分离可以加速页面加载速度,因为前端性能瓶颈通常出现在后端。

当然,前后端分离设计也存在一些挑战和限制,例如:

  1. 开发成本:前后端分离需要更多的开发人员和技术支持,因此可能会增加开发成本。
  2. 部署复杂性:前后端分离需要更多的配置和管理工作,因此可能会增加部署复杂性。
  3. 维护成本:前后端分离可能会增加维护成本,因为需要更多的人员来维护不同的代码版本和平台。

2 Go+Vue前后端分离

Go+Vue前后端分离也十分类似于Spring Boot+Vue前后端分离,因为他们本质上都是属于前后端分离设计,而且都有一个相同的约定,那就是Rustful的API风格和JSON数据格式进行数据传输,因此对于前端来说都是一样的接口,所以本次使用相同的前端代码,使用Go的Gin框架来重构后端代码,总体上也可分为以下大致几步:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VgRNDEfC-1682857177424)(一文搞懂Go+Vue前后端分离设计实践.assets/image-20230430195713075.png)]

Go后端需要的依赖如下:

github.com/gin-gonic/gin v1.9.0
github.com/go-sql-driver/mysql v1.7.1
github.com/spf13/cast v1.5.0

3 部分源码分享

3.1 加载配置

首先进行对MySQL连接驱动的初始化操作

config.go

var Db *sql.DB

const (
	DbDriver   = "mysql"
	DbUserName = "root"
	DbPasswd   = "12345"
	DbHost     = "127.0.0.1"
	DbPort     = 3306
	DbName     = "test"
)

func InitDB() {
	var err error
	Db, err = sql.Open(DbDriver, fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?charset=utf8",
		DbUserName, DbPasswd, DbHost, DbPort, DbName))
	if err != nil {
		panic(err)
	}
	Db.SetConnMaxLifetime(100 * time.Second)
	Db.SetMaxOpenConns(10)
	Db.SetMaxIdleConns(16)
}

3.2 数据模型

因为Go语言非常的简介,因此我们就将数据模型和对数据模型的数据库操作放在一个文件中

student.go

type Student struct {
	Id   int32  `json:"id"`
	Name string `json:"name"`
	Age  int8   `json:"age"`
}

var DefaultStudentDb *StudentDb

func NewStudentDb() *StudentDb {
	return &StudentDb{TbName: "student", Db: config.Db}
}

type StudentDb struct {
	TbName string
	Db     *sql.DB
}

func (s *StudentDb) Save(student Student) error {
	stmt, err := s.Db.Prepare(fmt.Sprintf("INSERT INTO %s(name,age) VALUES(?,?)", s.TbName))
	if err != nil {
		return err
	}
	if _, err = stmt.Exec(student.Name, student.Age); err != nil {
		return err
	}
	return nil
}

func (s *StudentDb) GetAll() ([]Student, error) {
	students := make([]Student, 0)
	rows, err := s.Db.Query(fmt.Sprintf("SELECT id,name,age from %s", s.TbName))
	if err != nil {
		return nil, err
	}
	for rows.Next() {
		var student Student
		if err := rows.Scan(&student.Id, &student.Name, &student.Age); err != nil {
			return nil, err
		}
		students = append(students, student)
	}
	return students, nil
}

func (s *StudentDb) GetOne(id int32) (Student, error) {
	var student Student
	rows, err := s.Db.Query(fmt.Sprintf("SELECT id,name,age from %s WHERE id=%d", s.TbName, id))
	if err != nil {
		return student, err
	}
	if rows.Next() {
		if err := rows.Scan(&student.Id, &student.Name, &student.Age); err != nil {
			return student, err
		}
	}
	return student, nil
}

func (s *StudentDb) Remove(id int32) error {
	stmt, err := s.Db.Prepare(fmt.Sprintf("DELETE FROM %s WHERE id=%d", s.TbName, id))
	if err != nil {
		return err
	}
	if _, err := stmt.Exec(); err != nil {
		return err
	}
	return nil
}

func (s *StudentDb) Update(student Student) error {
	stmt, err := s.Db.Prepare(fmt.Sprintf("UPDATE %s SET name=?,age=? WHERE id=?", s.TbName))
	if err != nil {
		return err
	}
	if _, err := stmt.Exec(student.Name, student.Age, student.Id); err != nil {
		return err
	}
	return nil
}

3.3 API设计

基于Gin框架的HTTP Rustful API设计

student.go

func Update(c *gin.Context) {
	student := model.Student{}
	if err := c.BindJSON(&student); err != nil {
		fmt.Println(err)
		c.JSON(http.StatusInternalServerError, -1)
		return
	}
	if err := model.DefaultStudentDb.Update(student); err != nil {
		c.JSON(http.StatusInternalServerError, -1)
		return
	}
	c.JSON(http.StatusOK, 1)
}

func Remove(c *gin.Context) {
	id := c.Param("id")
	if err := model.DefaultStudentDb.Remove(cast.ToInt32(id)); err != nil {
		c.JSON(http.StatusInternalServerError, -1)
		return
	}
	c.JSON(http.StatusOK, 1)
}

func FindById(c *gin.Context) {
	id := c.Param("id")
	student, err := model.DefaultStudentDb.GetOne(cast.ToInt32(id))
	if err != nil {
		c.JSON(http.StatusInternalServerError, -1)
		return
	}
	c.JSON(http.StatusOK, student)
}

func FindAll(c *gin.Context) {
	students, err := model.DefaultStudentDb.GetAll()
	if err != nil {
		c.JSON(http.StatusInternalServerError, -1)
		return
	}
	c.JSON(http.StatusOK, students)
}

func Save(c *gin.Context) {
	student := model.Student{}
	if err := c.BindJSON(&student); err != nil {
		c.JSON(http.StatusInternalServerError, -1)
		return
	}
	if err := model.DefaultStudentDb.Save(student); err != nil {
		c.JSON(http.StatusInternalServerError, -1)
		return
	}
	c.JSON(http.StatusOK, 1)
}

router.go

const (
	Port = 8081
)

func RunHttp() error {
	r := gin.Default()
	r.Use(CorsConfig())
	router := r.Group("/student")
	{
		router.POST("/save", Save)
		router.GET("/findAll", FindAll)
		router.GET("/findById/:id", FindById)
		router.POST("/update", Update)
		router.DELETE("/remove/:id", Remove)
	}
	return r.Run(fmt.Sprintf("127.0.0.1:%d", Port))
}

//解决跨域
func CorsConfig() gin.HandlerFunc {
	return func(c *gin.Context) {
		c.Header("Access-Control-Allow-Origin", "*") // 可将将 * 替换为指定的域名
		c.Header("Access-Control-Allow-Methods", "POST, GET, OPTIONS, PUT, DELETE, UPDATE")
		c.Header("Access-Control-Allow-Headers", "*")
		c.Header("Access-Control-Expose-Headers", "*")
		c.Header("Access-Control-Allow-Credentials", "true")
		c.Writer.Header().Set("Access-Control-Max-Age", "86400")
		if c.Request.Method == http.MethodOptions {
			c.AbortWithStatus(200)
		} else {
			c.Next()
		}
	}
}

3.4 启动文件

main.go

package main

import (
	"back_go/config"
	"back_go/model"
	"back_go/web"
)

func main() {
	config.InitDB()
	model.DefaultStudentDb = model.NewStudentDb()
	web.RunHttp()
}

4 总结和注意点

大家如果看过上一篇文章《手把手教你搭建Spring Boot+Vue前后端分离》的话相信你会感觉到Go相比Java在编码层面确实简洁了不少,也正是因此我才一点点的喜欢上Go语言,但是Java强大的生态体系是很难撼动的,就比如在使用Spring Boot搭建后端服务的时候,针对配置文件的读取、ORM框架的整合等等,Go语言的相关框架的整合确实不是十分的方便,当然如果要实现非常简单的代码,没有框架就是最好的框架。

在Go后端项目的配置方面,config.go文件中可以配置数据库相关信息,router.go文件中可以配置HTTP启动端口号相关信息,当然也可以写到相关的配置文件中,由Viper等工具进行解析。

5 代码获取方式

关于代码获取方式目前已经上传到GitHub,大家可以关注公众号【扯编程的淡】回复关键字【前后端】获取:

在这里插入图片描述

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。