当前位置: 首页 > news >正文

深入浅出:Gin框架-简介与API开发入门

深入浅出:Gin框架-简介与API开发入门

引言

Gin框架是基于Go语言的HTTP Web框架,凭借其简单易用、性能卓越和丰富的功能,成为构建高性能Web应用的理想选择。本文将深入浅出地介绍Gin框架的基础知识,并通过一个简单的案例,帮助您快速上手并开发一个功能完善的API。

什么是Gin框架?

Gin框架具有以下优势:

  • 高性能Gin框架的性能非常出色,尤其适合构建高并发的Web应用。
  • 简洁的APIGin框架提供了简洁且直观的API,使得路由定义、请求处理等操作变得非常简单。
  • 内置中间件Gin框架内置了日志记录、错误恢复等常用中间件,方便开发者快速搭建应用。
  • 良好的社区支持Gin框架拥有活跃的社区,提供了丰富的文档和插件,帮助开发者解决问题。

安装Gin框架

1. 创建一个新的Go项目

在终端中创建一个新的项目目录,并初始化Go模块:

mkdir my-gin-api
cd my-gin-api
go mod init my-gin-api

2. 安装Gin框架

接下来,使用go get命令安装Gin框架

go get -u github.com/gin-gonic/gin

创建第一个Gin框架应用

1. 编写主程序

在项目根目录下创建一个main.go文件,并编写以下代码:

package mainimport ("github.com/gin-gonic/gin"
)func main() {// 创建一个新的Gin路由器r := gin.Default()// 定义一个简单的GET路由r.GET("/", func(c *gin.Context) {c.JSON(200, gin.H{"message": "Hello, World!",})})// 启动HTTP服务器,监听8080端口r.Run(":8080")
}

2. 运行应用

在终端中运行以下命令启动应用:

go run main.go

打开浏览器,访问http://localhost:8080,您应该会看到如下输出:

{"message": "Hello, World!"
}

恭喜!您已经成功创建并运行了一个简单的Gin框架应用。

路由与HTTP请求处理

Gin框架提供了非常简洁的路由定义方式,支持常见的HTTP方法(如GET、POST、PUT、DELETE等)。我们可以通过以下几种方式来处理不同的HTTP请求。

1. 基本路由

您可以使用r.GETr.POSTr.PUTr.DELETE等方法来定义不同HTTP方法的路由。例如:

r.GET("/hello", func(c *gin.Context) {c.String(200, "Hello, Gin!")
})r.POST("/submit", func(c *gin.Context) {// 处理POST请求c.JSON(200, gin.H{"status": "success",})
})

2. 路径参数

Gin框架支持路径参数,您可以使用c.Param方法获取路径中的参数。例如:

r.GET("/user/:id", func(c *gin.Context) {id := c.Param("id")c.JSON(200, gin.H{"user_id": id,})
})

3. 查询参数

Gin框架也支持查询参数,您可以使用c.Query方法获取URL中的查询参数。例如:

r.GET("/search", func(c *gin.Context) {keyword := c.Query("keyword")c.JSON(200, gin.H{"keyword": keyword,})
})

4. 表单数据

对于表单提交的数据,您可以使用c.PostFormc.ShouldBind方法进行绑定。例如:

r.POST("/form", func(c *gin.Context) {var form struct {Name  string `form:"name" binding:"required"`Email string `form:"email" binding:"required,email"`}if err := c.ShouldBind(&form); err != nil {c.JSON(400, gin.H{"error": err.Error()})return}c.JSON(200, gin.H{"name":  form.Name,"email": form.Email,})
})

中间件

中间件是在请求到达最终处理函数之前或之后执行的函数,可以用于日志记录、认证、错误处理等。Gin框架内置了一些常用的中间件,您也可以自定义中间件。

1. 使用内置中间件

Gin框架内置了LoggerRecovery两个常用的中间件,分别用于日志记录和错误恢复。您可以在创建路由器时直接使用它们:

r := gin.New()
r.Use(gin.Logger())
r.Use(gin.Recovery())

2. 自定义中间件

您还可以创建自定义中间件。例如,创建一个简单的日志中间件:

func LoggerMiddleware() gin.HandlerFunc {return func(c *gin.Context) {start := time.Now()path := c.Request.URL.Pathraw := c.Request.URL.RawQueryc.Next()end := time.Now()latency := end.Sub(start)log.Printf("%s %s %s %d %v",c.ClientIP(),c.Request.Method,path,c.Writer.Status(),latency,)}
}r := gin.New()
r.Use(LoggerMiddleware())

JSON响应与错误处理

Gin框架提供了多种方式来处理响应和错误,最常见的是使用c.JSONc.Error方法。

1. JSON响应

您可以使用c.JSON方法发送JSON格式的响应。例如:

r.GET("/data", func(c *gin.Context) {data := map[string]interface{}{"name":  "John","age":   30,"email": "john@example.com",}c.JSON(200, data)
})

2. 错误处理

当发生错误时,您可以使用c.Error方法记录错误,并返回适当的HTTP状态码。例如:

r.GET("/error", func(c *gin.Context) {c.Error(errors.New("something went wrong"))c.AbortWithStatus(500)
})

数据库集成

Gin框架可以轻松集成各种数据库,最常用的是使用GORM作为ORM工具。GORM是一个功能强大的ORM库,支持多种数据库(如MySQL、PostgreSQL、SQLite等)。

1. 安装GORM

首先,安装GORM及其对应的数据库驱动:

go get -u gorm.io/gorm
go get -u gorm.io/driver/sqlite

2. 配置数据库连接

main.go中配置数据库连接:

import ("gorm.io/driver/sqlite""gorm.io/gorm"
)var db *gorm.DB
var err errorfunc init() {db, err = gorm.Open(sqlite.Open("test.db"), &gorm.Config{})if err != nil {panic("failed to connect database")}// 自动迁移模型db.AutoMigrate(&User{})
}type User struct {ID       uint   `gorm:"primaryKey"`Name     stringEmail    stringPassword string
}

3. CRUD操作

使用GORM可以轻松实现CRUD操作。例如:

// 创建新用户
func createUser(c *gin.Context) {var user Userif err := c.ShouldBindJSON(&user); err != nil {c.JSON(400, gin.H{"error": err.Error()})return}if err := db.Create(&user).Error; err != nil {c.JSON(500, gin.H{"error": err.Error()})return}c.JSON(201, user)
}// 获取所有用户
func getUsers(c *gin.Context) {var users []Userif err := db.Find(&users).Error; err != nil {c.JSON(500, gin.H{"error": err.Error()})return}c.JSON(200, users)
}// 更新用户信息
func updateUser(c *gin.Context) {id := c.Param("id")var user Userif err := db.First(&user, id).Error; err != nil {c.JSON(404, gin.H{"error": "user not found"})return}if err := c.ShouldBindJSON(&user); err != nil {c.JSON(400, gin.H{"error": err.Error()})return}if err := db.Save(&user).Error; err != nil {c.JSON(500, gin.H{"error": err.Error()})return}c.JSON(200, user)
}// 删除用户
func deleteUser(c *gin.Context) {id := c.Param("id")var user Userif err := db.Delete(&user, id).Error; err != nil {c.JSON(500, gin.H{"error": err.Error()})return}c.JSON(204, nil)
}

认证与授权

为了保护API的安全性,通常需要实现用户认证和授权。Gin框架可以轻松集成JWT(JSON Web Token)进行认证。

1. 安装JWT库

首先,安装JWT库:

go get -u github.com/golang-jwt/jwt/v4

2. 实现JWT认证

middleware/auth.go中实现JWT认证中间件:

package middlewareimport ("fmt""time""github.com/gin-gonic/gin""github.com/golang-jwt/jwt/v4"
)var jwtKey = []byte("my_secret_key")type Claims struct {Username string `json:"username"`jwt.RegisteredClaims
}func GenerateToken(username string) (string, error) {expirationTime := time.Now().Add(24 * time.Hour)claims := &Claims{Username: username,RegisteredClaims: jwt.RegisteredClaims{ExpiresAt: jwt.NewNumericDate(expirationTime),},}token := jwt.NewWithClaims(jwt.SigningMethodHS256, claims)return token.SignedString(jwtKey)
}func AuthMiddleware() gin.HandlerFunc {return func(c *gin.Context) {authHeader := c.GetHeader("Authorization")if authHeader == "" {c.JSON(401, gin.H{"error": "authorization header required"})c.Abort()return}tokenString := authHeader[len("Bearer "):]token, err := jwt.ParseWithClaims(tokenString, &Claims{}, func(token *jwt.Token) (interface{}, error) {return jwtKey, nil})if err != nil || !token.Valid {c.JSON(401, gin.H{"error": "invalid token"})c.Abort()return}claims, ok := token.Claims.(*Claims)if !ok {c.JSON(401, gin.H{"error": "invalid token claims"})c.Abort()return}c.Set("username", claims.Username)c.Next()}
}

3. 应用认证中间件

main.go中应用认证中间件:

r := gin.Default()auth := middleware.AuthMiddleware()r.POST("/login", func(c *gin.Context) {var login struct {Username string `json:"username" binding:"required"`Password string `json:"password" binding:"required"`}if err := c.ShouldBindJSON(&login); err != nil {c.JSON(400, gin.H{"error": err.Error()})return}// 简单的用户名和密码验证(实际应用中应使用更安全的方式)if login.Username == "admin" && login.Password == "password" {token, err := middleware.GenerateToken(login.Username)if err != nil {c.JSON(500, gin.H{"error": "failed to generate token"})return}c.JSON(200, gin.H{"token": token})} else {c.JSON(401, gin.H{"error": "invalid credentials"})}
})protected := r.Group("/api")
protected.Use(auth)
{protected.GET("/profile", func(c *gin.Context) {username := c.GetString("username")c.JSON(200, gin.H{"username": username})})
}

案例:使用Gin框架构建任务管理API

为了更好地理解Gin框架的实际应用,我们将通过一个简单的任务管理API来演示如何使用Gin框架构建一个完整的API。

1. 项目结构

创建以下文件和目录结构:

task-manager-api/
├── main.go
├── models/
│   └── task.go
├── routes/
│   └── task_routes.go
├── controllers/
│   └── task_controller.go
├── middleware/
│   └── auth.go
└── go.mod

2. 定义模型

models/task.go中定义任务模型:

package modelsimport ("time""gorm.io/gorm"
)type Task struct {ID          uint      `gorm:"primaryKey" json:"id"`Title       string    `json:"title" binding:"required"`Description string    `json:"description"`Status      string    `json:"status" binding:"oneof=pending completed canceled"` // 可选值: "pending", "completed", "canceled"CreatedAt   time.Time `json:"created_at"`UpdatedAt   time.Time `json:"updated_at"`
}

3. 定义控制器

controllers/task_controller.go中实现任务的CRUD操作:

package controllersimport ("net/http""strconv""github.com/gin-gonic/gin""gorm.io/gorm""task-manager-api/models"
)type TaskController struct {DB *gorm.DB
}func (tc *TaskController) GetTasks(c *gin.Context) {var tasks []models.Taskif err := tc.DB.Find(&tasks).Error; err != nil {c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})return}c.JSON(http.StatusOK, tasks)
}func (tc *TaskController) CreateTask(c *gin.Context) {var input models.Taskif err := c.ShouldBindJSON(&input); err != nil {c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})return}input.Status = "pending"if err := tc.DB.Create(&input).Error; err != nil {c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})return}c.JSON(http.StatusCreated, input)
}func (tc *TaskController) GetTask(c *gin.Context) {id, err := strconv.Atoi(c.Param("id"))if err != nil {c.JSON(http.StatusBadRequest, gin.H{"error": "Invalid task ID"})return}var task models.Taskif err := tc.DB.First(&task, id).Error; err != nil {c.JSON(http.StatusNotFound, gin.H{"error": "Task not found"})return}c.JSON(http.StatusOK, task)
}func (tc *TaskController) UpdateTask(c *gin.Context) {id, err := strconv.Atoi(c.Param("id"))if err != nil {c.JSON(http.StatusBadRequest, gin.H{"error": "Invalid task ID"})return}var task models.Taskif err := tc.DB.First(&task, id).Error; err != nil {c.JSON(http.StatusNotFound, gin.H{"error": "Task not found"})return}var input models.Taskif err := c.ShouldBindJSON(&input); err != nil {c.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})return}task.Title = input.Titletask.Description = input.Descriptiontask.Status = input.Statusif err := tc.DB.Save(&task).Error; err != nil {c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})return}c.JSON(http.StatusOK, task)
}func (tc *TaskController) DeleteTask(c *gin.Context) {id, err := strconv.Atoi(c.Param("id"))if err != nil {c.JSON(http.StatusBadRequest, gin.H{"error": "Invalid task ID"})return}var task models.Taskif err := tc.DB.Delete(&task, id).Error; err != nil {c.JSON(http.StatusInternalServerError, gin.H{"error": err.Error()})return}c.JSON(http.StatusNoContent, nil)
}

4. 定义路由

routes/task_routes.go中定义任务相关的路由:

package routesimport ("task-manager-api/controllers""github.com/gin-gonic/gin"
)func SetupTaskRoutes(r *gin.Engine, tc *controllers.TaskController) {r.GET("/tasks", tc.GetTasks)r.POST("/tasks", tc.CreateTask)r.GET("/tasks/:id", tc.GetTask)r.PUT("/tasks/:id", tc.UpdateTask)r.DELETE("/tasks/:id", tc.DeleteTask)
}

5. 主程序

main.go中集成所有部分:

package mainimport ("task-manager-api/controllers""task-manager-api/middleware""task-manager-api/routes""github.com/gin-gonic/gin""gorm.io/driver/sqlite""gorm.io/gorm"
)var db *gorm.DBfunc init() {var err errordb, err = gorm.Open(sqlite.Open("tasks.db"), &gorm.Config{})if err != nil {panic("failed to connect database")}// 自动迁移模型db.AutoMigrate(&controllers.Task{})
}func main() {r := gin.Default()// 注册任务控制器taskController := &controllers.TaskController{DB: db}// 设置任务路由routes.SetupTaskRoutes(r, taskController)// 设置认证中间件auth := middleware.AuthMiddleware()protected := r.Group("/api")protected.Use(auth){routes.SetupTaskRoutes(protected, taskController)}// 启动HTTP服务器,监听8080端口r.Run(":8080")
}

结语

通过本文,我们介绍了Gin框架的基础知识,并通过一个简单的任务管理API案例,展示了如何使用Gin框架快速开发一个功能完善的API。希望这篇文章能帮助您更好地理解和使用Gin框架

参考资料

  1. Gin官方文档
  2. GORM官方文档
  3. JWT官方文档
  4. Go官方文档
  5. Gin GitHub仓库

相关文章:

深入浅出:Gin框架-简介与API开发入门

深入浅出:Gin框架-简介与API开发入门 引言 Gin框架是基于Go语言的HTTP Web框架,凭借其简单易用、性能卓越和丰富的功能,成为构建高性能Web应用的理想选择。本文将深入浅出地介绍Gin框架的基础知识,并通过一个简单的案例&#xf…...

MySQL各种锁详解

什么是锁? 1.1 锁的解释 计算机协调多个进程或线程并发访问某一资源的机制。 1.2 锁的重要性 在数据库中,除传统计算资源(CPU、RAM、I/O等)的争抢,数据也是一种供多用户共享的资源。 如何保证数据并发访问的一致性&…...

海外的bug-hunters,不一样的403bypass

一种绕过403的新技术,跟大家分享一下。研究HTTP协议已经有一段时间了。发现HTTP协议的1.0版本可以绕过403。于是开始对lyncdiscover.microsoft.com域做FUZZ并且发现了几个403Forbidden的文件。 (访问fsip.svc为403) 在经过尝试后&#xff0…...

React 组件中 State 的定义、使用及正确更新方式

​🌈个人主页:前端青山 🔥系列专栏:React篇 🔖人终将被年少不可得之物困其一生 依旧青山,本期给大家带来React篇专栏内容React 组件中 State 的定义、使用及正确更新方式 前言 在 React 应用开发中,state …...

Jenkins 的HTTP Request 插件为什么不能配置Basic认证了

本篇遇到的问题 还是因为Jenkins需要及其所在的OS需要升级,升级策略是在一台新服务器上安装和配置最新版本的Jenkins, 当前的最新版本是: 2.479.2 LTS。 如果需要这个版本的话可以在官方站点下载,也可以到如下地址下载&#xff1…...

8 Bellman Ford算法SPFA

图论 —— 最短路 —— Bellman-Ford 算法与 SPFA_通信网理论基础 分别使用bellman-ford算法和dijkstra算法的应用-CSDN博客 图解Bellman-Ford计算过程以及正确性证明 - 知乎 (zhihu.com) 语雀版本 1 概念 **适用场景:**单源点,可以有负边&#xff0…...

nginx不允许静态文件被post请求显示405 not allowed

在单独站点的配置文件中 添加error_page 405 200 $request_uri; 即可!...

【c++笔试强训】(第三十二篇)

目录 数组变换(贪⼼位运算) 题目解析 讲解算法原理 编写代码 装箱问题(动态规划-01背包) 题目解析 讲解算法原理 编写代码 数组变换(贪⼼位运算) 题目解析 1.题目链接:数组变换__牛客网…...

shell脚本实战案例

文章目录 实战第一坑功能说明脚本实现 实战第一坑 实战第一坑:在Windows系统写了一个脚本,比如上面,随后上传到服务,执行会报错 原因: 解决方案:在linux系统touch文件,并通过vim添加内容&…...

OpenCV-图像阈值

简单阈值法 此方法是直截了当的。如果像素值大于阈值,则会被赋为一个值(可能为白色),否则会赋为另一个值(可能为黑色)。使用的函数是 cv.threshold。第一个参数是源图像,它应该是灰度图像。第二…...

lvgl9 Line(lv_line) 控件使用指南

文章目录 前言主体1. **Line 控件概述**2. **使用场景**3. **控件的样式**4. **设置点**5. **自动大小**6. **y 坐标反转**7. **事件处理**8. **示例代码** 总结 前言 在图形界面设计中,直线绘制是非常常见且重要的功能之一,尤其是在需要进行图形表示、…...

区块链概念 Web 3.0 实操

1. Web 3.0 概述 1.1 定义与背景 Web 3.0,也称为第三代互联网,是一个新兴的概念,它代表着互联网的未来发展和演进方向。Web 3.0的核心理念是去中心化、用户主权和智能化。这一概念的提出,旨在解决Web 2.0时代中用户数据隐私泄露…...

【人工智能】大数据平台技术及应用

文章目录 前言一、大数据平台基本概念及发展趋势1、数据量爆发式增长,发数据蓬勃发展2、大数据到底是什么?3、大数据处理与传统数据处理的差异4、为什么要建立大数据平台?5、大数据平台开源架构-Hadoop6、华为云大数据平台架构 二、大数据技术…...

Scala的模式匹配(7)

package hfdobject Test35 {case class Person(name:String)case class Student(name:String,className:String)//match case 能根据 类名和属性的信息,匹配到对应的类//注意://1 匹配的时候,case class的属性个数要对上//2 数学名不需要一一…...

使用 MATLAB 绘制三维散点图:根据坐标和距离映射点的颜色和大小

在数据可视化中,三维散点图是一种非常直观的方式来展示数据的分布。MATLAB 提供了强大的 scatter3 函数,可以用来绘制三维散点图,而通过调整点的颜色和大小,可以进一步增强图形的表现力。 在本篇博客中,我们将逐步讲解…...

数仓技术hive与oracle对比(五)

附录说明 附录是对测试过程中涉及到的一些操作进行记录和解析。 oracle清除缓存 alter system flush shared_pool; 将使library cache和data dictionary cache以前保存的sql执行计划全部清空,但不会清空共享sql区或者共享pl/sql区里面缓存的最近被执行的条目。刷…...

金融数学在股市交易中的具体应用

### 1. 风险管理 - **VaR(在险价值)**: VaR是衡量投资组合潜在损失的指标。例如,如果一个投资组合的VaR为100万元,置信水平为95%,这意味着在未来的一个交易日内,有95%的可能性该投资组合的损失不会超过100…...

Spring6:1 概述

Spring6:1 概述 标签 JAVASpring 目录 Spring 是什么?Spring 的狭义和广义 广义的 Spring:Spring 技术栈狭义的 Spring:Spring Framework Spring Framework 特点Spring 模块组成Spring6 特点 版本要求本课程软件版本 1. 概述 …...

Python Selenium 各浏览器驱动下载与配置使用(详细流程)

1、安装 pip install selenium 2、浏览器驱动下载 Chrome(google)浏览器驱动 下载地址:http://chromedriver.storage.googleapis.com/index.html 或 https://sites.google.com/a/chromium.org/chromedriver/home . 下载地址:http://chromedriver.stor…...

C语言期末考试——重点考点

目录 1.C语言的结构 2.三种循环结构 3.逻辑真假判断 4. printf函数 5. 强制类型转化 6. 多分支选择结构 7. 标识符的定义 8. 三目运算符 1.C语言的结构 选择结构、顺序结构、循环结构 2.三种循环结构 for、while、do-while 3.逻辑真假判断 C语言用0表示false,用非0(不…...

ES6从入门到精通:前言

ES6简介 ES6(ECMAScript 2015)是JavaScript语言的重大更新,引入了许多新特性,包括语法糖、新数据类型、模块化支持等,显著提升了开发效率和代码可维护性。 核心知识点概览 变量声明 let 和 const 取代 var&#xf…...

Spark 之 入门讲解详细版(1)

1、简介 1.1 Spark简介 Spark是加州大学伯克利分校AMP实验室(Algorithms, Machines, and People Lab)开发通用内存并行计算框架。Spark在2013年6月进入Apache成为孵化项目,8个月后成为Apache顶级项目,速度之快足见过人之处&…...

【人工智能】神经网络的优化器optimizer(二):Adagrad自适应学习率优化器

一.自适应梯度算法Adagrad概述 Adagrad(Adaptive Gradient Algorithm)是一种自适应学习率的优化算法,由Duchi等人在2011年提出。其核心思想是针对不同参数自动调整学习率,适合处理稀疏数据和不同参数梯度差异较大的场景。Adagrad通…...

java 实现excel文件转pdf | 无水印 | 无限制

文章目录 目录 文章目录 前言 1.项目远程仓库配置 2.pom文件引入相关依赖 3.代码破解 二、Excel转PDF 1.代码实现 2.Aspose.License.xml 授权文件 总结 前言 java处理excel转pdf一直没找到什么好用的免费jar包工具,自己手写的难度,恐怕高级程序员花费一年的事件,也…...

【SQL学习笔记1】增删改查+多表连接全解析(内附SQL免费在线练习工具)

可以使用Sqliteviz这个网站免费编写sql语句,它能够让用户直接在浏览器内练习SQL的语法,不需要安装任何软件。 链接如下: sqliteviz 注意: 在转写SQL语法时,关键字之间有一个特定的顺序,这个顺序会影响到…...

什么?连接服务器也能可视化显示界面?:基于X11 Forwarding + CentOS + MobaXterm实战指南

文章目录 什么是X11?环境准备实战步骤1️⃣ 服务器端配置(CentOS)2️⃣ 客户端配置(MobaXterm)3️⃣ 验证X11 Forwarding4️⃣ 运行自定义GUI程序(Python示例)5️⃣ 成功效果![在这里插入图片描述](https://i-blog.csdnimg.cn/direct/55aefaea8a9f477e86d065227851fe3d.pn…...

C++ Visual Studio 2017厂商给的源码没有.sln文件 易兆微芯片下载工具加开机动画下载。

1.先用Visual Studio 2017打开Yichip YC31xx loader.vcxproj,再用Visual Studio 2022打开。再保侟就有.sln文件了。 易兆微芯片下载工具加开机动画下载 ExtraDownloadFile1Info.\logo.bin|0|0|10D2000|0 MFC应用兼容CMD 在BOOL CYichipYC31xxloaderDlg::OnIni…...

2023赣州旅游投资集团

单选题 1.“不登高山,不知天之高也;不临深溪,不知地之厚也。”这句话说明_____。 A、人的意识具有创造性 B、人的认识是独立于实践之外的 C、实践在认识过程中具有决定作用 D、人的一切知识都是从直接经验中获得的 参考答案: C 本题解…...

Redis的发布订阅模式与专业的 MQ(如 Kafka, RabbitMQ)相比,优缺点是什么?适用于哪些场景?

Redis 的发布订阅(Pub/Sub)模式与专业的 MQ(Message Queue)如 Kafka、RabbitMQ 进行比较,核心的权衡点在于:简单与速度 vs. 可靠与功能。 下面我们详细展开对比。 Redis Pub/Sub 的核心特点 它是一个发后…...

C++:多态机制详解

目录 一. 多态的概念 1.静态多态(编译时多态) 二.动态多态的定义及实现 1.多态的构成条件 2.虚函数 3.虚函数的重写/覆盖 4.虚函数重写的一些其他问题 1).协变 2).析构函数的重写 5.override 和 final关键字 1&#…...