Go语言24小时极速学习教程(五)Go语言中的SpringMVC框架——Gin
作为一个真正能用的企业级应用,怎么能缺少RESTful接口呢?所以我们需要尝试在Go语言环境中写出我们的对外接口,这样前端就可以借由Gin框架访问我们数据库中的数据了。
一、Gin框架的使用
1. 安装 Gin
首先,你需要在你的 Go 项目中安装 Gin 框架。可以使用 go get 命令来安装,如果你使用了GoLand作为IDE,直接照着提示引入即可:
go get -u github.com/gin-gonic/gin
2. 创建一个简单的 Gin 应用
package mainimport ("github.com/gin-gonic/gin""net/http"
)func main() {// 创建一个默认的路由引擎r := gin.Default()// 定义一个 GET 请求的路由r.GET("/", func(c *gin.Context) {c.String(http.StatusOK, "Hello, World!")})// 运行服务器,默认监听在 0.0.0.0:8080r.Run()
}
3. 路由定义
Gin 支持多种 HTTP 方法,如 GET、POST、PUT、DELETE 等。以下是一些基本的路由定义示例:
r.GET("/someGet", getting)
r.POST("/somePost", posting)
r.PUT("/somePut", putting)
r.DELETE("/someDelete", deleting)
r.PATCH("/somePatch", patching)
r.HEAD("/someHead", head)
r.OPTIONS("/someOptions", options)
类比于Java,相当于设置了GetMapping、PostMapping。
4. 路由参数
Gin 支持路径参数和查询参数。
路径参数
r.GET("/user/:name", func(c *gin.Context) {name := c.Param("name")c.String(http.StatusOK, "Hello %s", name)
})
查询参数
r.GET("/welcome", func(c *gin.Context) {firstname := c.DefaultQuery("firstname", "Guest")lastname := c.Query("lastname") // 等同于 c.Request.URL.Query().Get("lastname")c.String(http.StatusOK, "Hello %s %s", firstname, lastname)
})
5. 处理表单数据
Gin 可以轻松处理表单数据。
r.POST("/form_post", func(c *gin.Context) {message := c.PostForm("message")nick := c.DefaultPostForm("nick", "anonymous")c.JSON(200, gin.H{"status": "posted","message": message,"nick": nick,})
})
6. 处理 JSON 数据
Gin 可以轻松处理 JSON 请求和响应。
r.POST("/json", func(c *gin.Context) {var json struct {Name string `json:"name" binding:"required"`Age int `json:"age" binding:"required"`}if c.BindJSON(&json) == nil {c.JSON(http.StatusOK, gin.H{"name": json.Name,"age": json.Age,})} else {c.JSON(http.StatusBadRequest, gin.H{"error": "请求的JSON有误"})}
})
7. 中间件
Gin 支持中间件,可以在请求处理之前或之后执行一些操作,放在Java里就相当于AspectJ作为AOP层来实现切面编程。
func Logger() gin.HandlerFunc {return func(c *gin.Context) {t := time.Now()// 设置 example 变量c.Set("example", "12345")// 请求前c.Next()// 请求后latency := time.Since(t)log.Print(latency)// 访问发送的 statusstatus := c.Writer.Status()log.Println(status)}
}func main() {r := gin.New()r.Use(Logger())r.GET("/test", func(c *gin.Context) {example := c.MustGet("example").(string)// 打印:"12345"log.Println(example)})// 监听并在 0.0.0.0:8080 上启动服务r.Run(":8080")
}
8. 静态文件服务
Gin 可以轻松地提供静态文件服务。
r.Static("/assets", "./assets")
r.StaticFS("/more_static", http.Dir("my_file_system"))
r.StaticFile("/favicon.ico", "./resources/favicon.ico")
9. 分组路由
Gin 支持路由分组,可以更好地组织和管理路由。
v1 := r.Group("/v1")
{v1.POST("/login", loginEndpoint)v1.POST("/submit", submitEndpoint)v1.POST("/read", readEndpoint)
}v2 := r.Group("/v2")
{v2.POST("/login", loginEndpoint)v2.POST("/submit", submitEndpoint)v2.POST("/read", readEndpoint)
}
10. 错误处理
Gin 提供了方便的错误处理机制。
r.GET("/panic", func(c *gin.Context) {panic("啊哦,出错了")
})r.Use(gin.Recovery())
11. 自定义 HTTP 配置
你可以自定义 HTTP 服务器配置。
s := &http.Server{Addr: ":8080",Handler: r,ReadTimeout: 10 * time.Second,WriteTimeout: 10 * time.Second,MaxHeaderBytes: 1 << 20,
}
s.ListenAndServe()
12. 使用模板引擎
Gin 支持模板引擎来渲染 HTML 页面。
r.LoadHTMLGlob("templates/*")r.GET("/index", func(c *gin.Context) {c.HTML(http.StatusOK, "index.tmpl", gin.H{"title": "Main website",})
})
Gin本身没有模板引擎,它用的是Go自带的标准库html/template和text/template,下面是一个模板文件的例子:
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>{{ .title }}</title>
</head>
<body><h1>{{ .title }}</h1><p>欢迎光临我的网站!</p>
</body>
</html>
下面我单独开一章详细介绍一下模板如何编写
13. 使用 Session
Gin 本身不提供 Session 管理,但你可以使用第三方库如 gin-contrib/sessions。
go get github.com/gin-contrib/sessions
import ("github.com/gin-contrib/sessions""github.com/gin-contrib/sessions/cookie""github.com/gin-gonic/gin"
)func main() {r := gin.Default()store := cookie.NewStore([]byte("secret"))r.Use(sessions.Sessions("mysession", store))r.GET("/incr", func(c *gin.Context) {session := sessions.Default(c)var count intv := session.Get("count")if v == nil {count = 0} else {count = v.(int)count++}session.Set("count", count)session.Save()c.JSON(200, gin.H{"count": count})})r.Run(":8000")
}
14. 使用 WebSocket
Gin 可以与 WebSocket 结合使用。
import ("github.com/gin-gonic/gin""github.com/gorilla/websocket""net/http"
)var upgrader = websocket.Upgrader{CheckOrigin: func(r *http.Request) bool {return true},
}func websocketHandler(c *gin.Context) {conn, err := upgrader.Upgrade(c.Writer, c.Request, nil)if err != nil {return}defer conn.Close()for {mt, message, err := conn.ReadMessage()if err != nil {break}err = conn.WriteMessage(mt, message)if err != nil {break}}
}func main() {r := gin.Default()r.GET("/ws", websocketHandler)r.Run(":8080")
}
二、模板引擎的使用
刚才我们提到了模板引擎,一般来说对于需要进行SEO优化的网站才会用上模板引擎,而普通的业务系统往往都是前后端分离的,对于前后端分离项目则不需要研究模板引擎如何使用。此外,如果你作为架构师正在考虑生成DSL、生成代码这类操作,也可以使用模板引擎来完成目标。类比于Java而言,模板引擎非常像Freemarker、Velocity这样的框架。Go 语言的 html/template 包是一个功能强大的模板引擎,支持多种模板语法和功能。以下是 html/template 中常用的模板写法:
1. 基本语法
变量输出
使用 {{ .variable }} 输出变量的值。
<p>你好, {{ .Name }}</p>
条件判断
使用 {{ if .condition }} ... {{ else }} ... {{ end }} 进行条件判断。
{{ if .IsAdmin }}<p>管理员</p>
{{ else }}<p>用户</p>
{{ end }}
循环迭代
使用 {{ range .list }} ... {{ end }} 进行循环迭代。
<ul>
{{ range .Items }}<li>{{ . }}</li>
{{ end }}
</ul>
2. 模板继承
定义块
使用 {{ define "block_name" }} ... {{ end }} 定义块。
{{ define "content" }}<p>这是一个内容块</p>
{{ end }}
包含块
使用 {{ template "block_name" . }} 包含块。
{{ template "content" . }}
基础模板
使用 {{ define "base" }} ... {{ end }} 定义基础模板。
{{ define "base" }}
<!DOCTYPE html>
<html>
<head><meta charset="UTF-8"><title>{{ .Title }}</title>
</head>
<body><header><h1>{{ .Title }}</h1></header><main>{{ template "content" . }}</main><footer><p>© 2024 瑜美科技</p></footer>
</body>
</html>
{{ end }}
3. 模板函数
内置函数
html/template 提供了一些内置函数,如 and, or, not, len, index 等。
<p>总计数量: {{ len .Items }}</p>
自定义函数
使用 SetFuncMap 注册自定义函数。
package mainimport ("html/template""net/http""time"
)func formatAsDate(t time.Time) string {return t.Format("2006-01-02")
}func main() {tmpl := template.Must(template.New("").Funcs(template.FuncMap{"formatAsDate": formatAsDate,}).ParseFiles("templates/index.tmpl"))http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {tmpl.ExecuteTemplate(w, "index.tmpl", map[string]interface{}{"Date": time.Now(),})})http.ListenAndServe(":8080", nil)
}
模板文件中使用自定义函数:
<p>当前日期为: {{ formatAsDate .Date }}</p>
4. 模板注释
使用 {{/* ... */}} 添加模板注释。
{{/* 这是一段用户用F12也看不到的注释 */}}
<p>您好,{{ .Name }}</p>
5. 模板变量
使用 {{ $variable := .value }} 定义只在模板渲染生命周期过程中临时产生的变量。
{{ $name := .Name }}
<p>您好,{{ $name }}</p>
6. 模板安全
自动转义
html/template 自动转义 HTML 内容,防止 XSS 攻击。
<p>{{ .UnsafeHTML }}</p>
不转义
使用 {{ . | safeHTML }} 不转义 HTML 内容。
<p>{{ .UnsafeHTML | safeHTML }}</p>
总结
Gin 是一个功能强大且易于使用的 Web 框架,适用于构建高性能的 Web 应用。html/template 提供了丰富的模板语法和功能,包括变量输出、条件判断、循环迭代、模板继承、模板函数、模板注释、模板操作、模板变量和模板安全等。通过上述示例,你可以快速上手并开始构建你的 Go 语言 Web 应用。至此Go语言24小时极速学习教程就结束了,你可以直接开始参与一些企业级的Go Web项目了。怎么样,学的嗖嗖快对吧,至于数不熟练还需要靠练习,有很多编程中的技巧、数据结构、算法和设计模式与Java基本一致,所以如果你Java和C#玩的很6,相信Go语言也能在1天之内玩的非常6。
相关文章:
Go语言24小时极速学习教程(五)Go语言中的SpringMVC框架——Gin
作为一个真正能用的企业级应用,怎么能缺少RESTful接口呢?所以我们需要尝试在Go语言环境中写出我们的对外接口,这样前端就可以借由Gin框架访问我们数据库中的数据了。 一、Gin框架的使用 1. 安装 Gin 首先,你需要在你的 Go 项目…...
【汇编】c++游戏开发
由一起学编程创作的‘C/C项目实战:2D射击游戏开发(简易版), 440 行源码分享来啦~’: C/C项目实战:2D射击游戏开发(简易版), 440 行源码分享来啦~_射击c-CSDN博客文章浏览…...
Android Studio | 修改镜像地址为阿里云镜像地址,启动App
在项目文件的目录下的 settings.gradle.kts 中修改配置,配置中包含插件和依赖项 pluginManagement {repositories {maven { urluri ("https://www.jitpack.io")}maven { urluri ("https://maven.aliyun.com/repository/releases")}maven { urlu…...
Rocky linux8 安装php8.0
Rocky linux8 安装php8.0 1.安装remi源2.列出php版本3.变更php版本,Rocky8有提供php8版本,所以切换Rocky8提供的版本,而不是remi提供的版本,不过remi有提供php8.1和php8.2版本。4.切换成remi提供的8.0版本5.安装phpendl 1.安装rem…...
Ubuntu 18 EDK2 环境编译
视频:在全新的Ubuntu上从零搭建UEFI的EDK2开发环境 开始:git clone https://github.com/tianocore/edk2.git 开始编译BaseTools前先更新一下子模块:git submodule update --init ,然后:make -C BaseTools/ 问题1&a…...
C语言项⽬实践-贪吃蛇
目录 1.项目要点 2.窗口设置 2.1mode命令 2.2title命令 2.3system函数 2.Win32 API 2.1 COORD 2.2 GetStdHandle 2.3 CONSOLE_CURSOR_INFO 2.4 GetConsoleCursorInfo 2.5 SetConsoleCursorInfo 2.5 SetConsoleCursorPosition 2.7 GetAsyncKeyState 3.贪吃蛇游戏设…...
智慧安防丨以科技之力,筑起防范人贩的铜墙铁壁
近日,贵州省贵阳市中级人民法院对余华英拐卖儿童案做出了一审宣判,判处其死刑,剥夺政治权利终身,并处没收个人全部财产。这一判决不仅彰显了法律的威严,也再次唤起了社会对拐卖儿童犯罪的深切关注。 余华英自1993年至2…...
Spring:IoC/DI加载properties文件
Spring框架可以通过Spring的配置文件完成两个数据源druid和C3P0的配置(Spring:IOC/DI配置管理第三方bean),但是其中包含了一些问题,我们来分析下: 这两个数据源中都使用到了一些固定的常量如数据库连接四要素…...
Docker 篇-Docker 详细安装、了解和使用 Docker 核心功能(数据卷、自定义镜像 Dockerfile、网络)
🔥博客主页: 【小扳_-CSDN博客】 ❤感谢大家点赞👍收藏⭐评论✍ 文章目录 1.0 Docker 概述 1.1 Docker 主要组成部分 1.2 Docker 安装 2.0 Docker 常见命令 2.1 常见的命令介绍 2.2 常见的命令演示 3.0 数据卷 3.1 数据卷常见的命令 3.2 常见…...
深挖C++赋值
详解赋值 const int a 10; int b a;&a 0x000000b7c6afef34 {56496} &a 0x000000b7c6afef34 {10} 3. &b 0x000000b7c6afef54 {10} 总结: int a 10 是指在内存中(栈)中创建一个int (4 byte)大小的空间…...
【免越狱】iOS砸壳 可下载AppStore任意版本 旧版本IPA下载
软件介绍 下载iOS旧版应用,简化繁琐的抓包流程。 一键生成去更新IPA(手机安装后,去除App Store的更新检测)。 软件界面 支持系统 Windows 10/Windows 8/Windows 7(由于使用了Fiddler库,因此需要.Net环境…...
【python笔记02】面向对象思想
关于面向对象要学会啥? 面向对象编程思想面向对象基本概念 对象类 添加和获取对象属性魔术方法(三个常见的)面向对象案例 面向对象编程思想 两个时代的两个产物,没有好坏之分,小系统用面向过程,团队开发…...
Java基础-Java多线程机制
(创作不易,感谢有你,你的支持,就是我前行的最大动力,如果看完对你有帮助,请留下您的足迹) 目录 一、引言 二、多线程的基本概念 1. 线程与进程 2. 多线程与并发 3. 多线程的优势 三、Java多线程的实…...
MySQL技巧之跨服务器数据查询:基础篇-A数据库与B数据库查询合并--封装到存储过程中
MySQL技巧之跨服务器数据查询:基础篇-A数据库与B数据库查询合并–封装到存储过程中 我们的最终目的是什么?当然的自动执行这些合并操作! 上一篇 MySQL技巧之跨服务器数据查询:基础篇-A数据库与B数据库查询合并 我们已经知道怎么合…...
MATLAB向量元素的引用
我们定义一个向量后,如果想引用的话,可以通过索引 i n d ind ind来实现。 注意:MATLAB中向量的开始索引是1,与许多编程语言不同。 例如: 如果想引用多个的话,可以用索引 i n d ind ind来提取多个位置 例如…...
leetcode-44-通配符匹配
题解: 代码: 参考: (1)牛客华为机试HJ71字符串通配符 (2)leetcode-10-正则表达式匹配...
基于YOLOv8深度学习的智慧课堂学生专注度检测系统(PyQt5界面+数据集+训练代码)
本研究提出了一种基于YOLOv8深度学习的智慧课堂学生专注度检测系统,旨在实现对课堂中学生专注度的实时分析与评估。随着智慧教育的快速发展,学生的课堂表现和专注度成为评估学习效果的重要因素之一。然而,传统的专注度评估方法往往依赖于主观…...
vue项目使用eslint+prettier管理项目格式化
代码格式化、规范化说明 使用eslintprettier进行格式化,vscode中需要安装插件ESLint、Prettier - Code formatter,且格式化程序选择为后者(vue文件、js文件要分别设置) 对于eslint规则,在格式化时不会全部自动调整&…...
Java基础-组件及事件处理(中)
(创作不易,感谢有你,你的支持,就是我前行的最大动力,如果看完对你有帮助,请留下您的足迹) 目录 BorderLayout布局管理器 说明: 示例: FlowLayout布局管理器 说明: …...
UNIX网络编程-TCP套接字编程(实战)
概述 TCP客户端/服务器程序示例是执行如下步骤的一个回射服务器: 客户端从标准输入读入一行文本,并写给服务器。服务器从网络输入读入这行文本,并回射给客户端。客户端从网络输入读入这行回射文本,并显示在标准输出上。 TCP服务器…...
汇编常见指令
汇编常见指令 一、数据传送指令 指令功能示例说明MOV数据传送MOV EAX, 10将立即数 10 送入 EAXMOV [EBX], EAX将 EAX 值存入 EBX 指向的内存LEA加载有效地址LEA EAX, [EBX4]将 EBX4 的地址存入 EAX(不访问内存)XCHG交换数据XCHG EAX, EBX交换 EAX 和 EB…...
Redis数据倾斜问题解决
Redis 数据倾斜问题解析与解决方案 什么是 Redis 数据倾斜 Redis 数据倾斜指的是在 Redis 集群中,部分节点存储的数据量或访问量远高于其他节点,导致这些节点负载过高,影响整体性能。 数据倾斜的主要表现 部分节点内存使用率远高于其他节…...
Web 架构之 CDN 加速原理与落地实践
文章目录 一、思维导图二、正文内容(一)CDN 基础概念1. 定义2. 组成部分 (二)CDN 加速原理1. 请求路由2. 内容缓存3. 内容更新 (三)CDN 落地实践1. 选择 CDN 服务商2. 配置 CDN3. 集成到 Web 架构 …...
技术栈RabbitMq的介绍和使用
目录 1. 什么是消息队列?2. 消息队列的优点3. RabbitMQ 消息队列概述4. RabbitMQ 安装5. Exchange 四种类型5.1 direct 精准匹配5.2 fanout 广播5.3 topic 正则匹配 6. RabbitMQ 队列模式6.1 简单队列模式6.2 工作队列模式6.3 发布/订阅模式6.4 路由模式6.5 主题模式…...
springboot整合VUE之在线教育管理系统简介
可以学习到的技能 学会常用技术栈的使用 独立开发项目 学会前端的开发流程 学会后端的开发流程 学会数据库的设计 学会前后端接口调用方式 学会多模块之间的关联 学会数据的处理 适用人群 在校学生,小白用户,想学习知识的 有点基础,想要通过项…...
4. TypeScript 类型推断与类型组合
一、类型推断 (一) 什么是类型推断 TypeScript 的类型推断会根据变量、函数返回值、对象和数组的赋值和使用方式,自动确定它们的类型。 这一特性减少了显式类型注解的需要,在保持类型安全的同时简化了代码。通过分析上下文和初始值,TypeSc…...
从面试角度回答Android中ContentProvider启动原理
Android中ContentProvider原理的面试角度解析,分为已启动和未启动两种场景: 一、ContentProvider已启动的情况 1. 核心流程 触发条件:当其他组件(如Activity、Service)通过ContentR…...
WPF八大法则:告别模态窗口卡顿
⚙️ 核心问题:阻塞式模态窗口的缺陷 原始代码中ShowDialog()会阻塞UI线程,导致后续逻辑无法执行: var result modalWindow.ShowDialog(); // 线程阻塞 ProcessResult(result); // 必须等待窗口关闭根本问题:…...
MySQL的pymysql操作
本章是MySQL的最后一章,MySQL到此完结,下一站Hadoop!!! 这章很简单,完整代码在最后,详细讲解之前python课程里面也有,感兴趣的可以往前找一下 一、查询操作 我们需要打开pycharm …...
【Veristand】Veristand环境安装教程-Linux RT / Windows
首先声明,此教程是针对Simulink编译模型并导入Veristand中编写的,同时需要注意的是老用户编译可能用的是Veristand Model Framework,那个是历史版本,且NI不会再维护,新版本编译支持为VeriStand Model Generation Suppo…...
