热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

Go语言实现Redis客户端与服务器的交互机制深入解析

在前文对Godisv1.0版本的基础功能进行了详细介绍后,本文将重点探讨如何实现客户端与服务器之间的交互机制。通过具体代码实现,使客户端与服务器能够顺利通信,赋予项目实际运行的能力。本文将详细解析Go语言在实现这一过程中的关键技术和实现细节,帮助读者深入了解Redis客户端与服务器的交互原理。

写在前面

在前一篇梳理了Godis v1.0版本的基本功能,这一篇要做的是实现客户端/服务端的交互。先让代码跑起来,才算有了生命力。
本篇Godis版本号:v0.0.1

在这个系列文章里,尽量减少介绍Golang语法、C语言语法和redis原理,聚焦在“用Golang实现Redis”的主题上。其中如有疏漏、不足,还请指正。

进入正题

Redis事件处理器

既要实现C/S交互,网络编程必不可少。在Redis中,有实现方式:

*Redis基于 Reactor 模式开发了自己的网络事件处理器: 这个处理器被称为文件事件处理器(file event handler):

  • 文件事件处理器使用 I/O 多路复用(multiplexing)程序来同时监听多个套接字, 并根据套接字目前执行的任务来为套接字关联不同的事件处理器。
  • 当被监听的套接字准备好执行连接应答(accept)、读取(read)、写入(write)、关闭(close)等操作时, 与操作相对应的文件事件就会产生, 这时文件事件处理器就会调用套接字之前关联好的事件处理器来处理这些事件。*

Redis事件处理器的架构图:图片描述

Redis兼顾了简单性和高性能,但出于对其代码复杂性的考虑,Godis v0.0.1会采用更简单的弱智方案。Godis的编码使用最简单的同步技术,争取在网络处理方面简单化,因为复杂的也不会:)。暂时舍弃高性能,只求可用。在未来的版本中,会结合Golang自身的特性,优化网络层。

Godis的方案

客户端/服务端的交互,只需使用Golang的net包进行socket编程,编写一个流程为监听-读取-处理-回复的服务端和一个流程为建立-发送-接收-显示的客户端即可。
图片描述
图中英文均为Golang中net包的主要函数,这里简要说明一下在这一篇中使用的net包函数:
ResolveTCPAddr :
用例:tcpAddr, err := net.ResolveTCPAddr("tcp4", “127.0.0.1:9736”);
功能:创建TCPAddr类型的数据结构,其中包含IP和Port,留作连接之用;

ListenTCP:
用例:netListen, err := net.Listen("tcp", "127.0.0.1:9763");
功能:监听

Accept:
用例:conn, err := netListen.Accept()
功能:接收请求

Read:
用例:n, err := conn.Read(buff)
功能:读取数据

Write:
用例:conn.Write([]byte(buff))
功能:响应数据

net.DialTCP:
用例:conn, err := net.DialTCP("tcp", nil, tcpAddr)
功能:建立连接

代码实现

按照前图的流程,服务端的部分代码如下所示:

func main() {netListen, err := net.Listen("tcp", "127.0.0.1:9736")if err != nil {log.Print("listen err ")}//checkError(err)defer netListen.Close()for {conn, err := netListen.Accept()if err != nil {continue}go handle(conn)}}

真实完整代码实现请见这里。
在建立客户端之前,我们通过telnet测试下服务端是否可用(Redis也支持Telnet方式连接、请求,后面的协议部分会介绍)。
编译godis-server.go
go build godis-server.go 并启动 ./godis-server

执行 telnet localhost 9736

可以看到回复如下:
图片描述

接下来,按照简单流程图的步骤,客户端的实现如下:

func main() {IPPort := "127.0.0.1:9736"reader := bufio.NewReader(os.Stdin)fmt.Println("Hi Godis")tcpAddr, err := net.ResolveTCPAddr("tcp4", IPPort)checkError(err)conn, err := net.DialTCP("tcp", nil, tcpAddr)checkError(err)defer conn.Close()for {fmt.Print(IPPort + "> ")text, _ := reader.ReadString('\n')//清除掉回车换行符text = strings.Replace(text, "\n", "", -1)send2Server(text, conn)buff := make([]byte, 1024)n, err := conn.Read(buff)checkError(err)if n == 0 {fmt.Println(IPPort+"> ", "nil")} else {fmt.Println(IPPort+">", string(buff))}}
}

编译cli.go 并启动./cli
与服务端交互,如下图所示:
图片描述

到这里只是实现了基本的客户端/服务端通信。接下来为它添加少许交互选项,也就是在执行Redis命令行时的选项,让它看起来更像是Redis。

clipboard.png

服务端添加平滑退出。

未来增加持久化后,会在平滑退出时持久化数据到磁盘,防止丢失。
平滑退出这里使用的方式是让客户端监听信号,有“退出”信号触达,做完收尾工作再退出。
简化代码如下:

func sigHandler(c chan os.Signal) {for s := range c {switch s {case syscall.SIGHUP, syscall.SIGINT, syscall.SIGTERM, syscall.SIGQUIT:exitHandler()default:fmt.Println("signal ", s)}}
}func exitHandler() {fmt.Println("exiting smoothly ...")fmt.Println("bye ")os.Exit(0)
}

编译之后,执行Ctrl+c退出,可以看到:

clipboard.png

本篇遇到过的问题:

1.服务端读取客户端信息时,发生err时没有return,导致服务端会一直读取数据失败。
解决方案:return或者其他可以退出goroutine的方案。
本篇就是这样了。简陋是简陋了点,又不是不能用,李姐万岁。完整代码请看本篇对应的release,(只有release才是当前本篇文章对应的完整代码,当前repo状态不是)。

下集预告

1.[实现get/set命令。][8]



推荐阅读
  • 在幼儿园中,有 \( n \) 个小朋友需要通过投票来决定是否午睡。尽管这个问题对每个孩子来说并不是特别重要,但他们仍然希望通过谦让的方式达成一致。每个人都有自己的偏好,但为了集体和谐,他们决定采用一种最小割的方法来解决这一问题。这种方法不仅能够确保每个人的意愿得到尽可能多的尊重,还能找到一个最优的解决方案,使整体满意度最大化。 ... [详细]
  • Go语言中Goroutine与通道机制及其异常处理深入解析
    在Go语言中,Goroutine可视为一种轻量级的并发执行单元,其资源消耗远低于传统线程,初始栈大小仅为2KB,而普通线程则通常需要几MB。此外,Goroutine的调度由Go运行时自动管理,能够高效地支持成千上万个并发任务。本文深入探讨了Goroutine的工作原理及其与通道(channel)的配合使用,特别是在异常处理方面的最佳实践,为开发者提供了一套完整的解决方案,以确保程序的稳定性和可靠性。 ... [详细]
  • 本文探讨了将PEBuilder转换为DIBooter.sh的方法,重点介绍了如何将DI工具集成到启动层,实现离线镜像引导安装。通过使用DD命令替代传统的grub-install工具,实现了GRUB的离线安装。此外,还详细解析了bootice工具的工作原理及其在该过程中的应用,确保系统在无网络环境下也能顺利引导和安装。 ... [详细]
  • voc生成xml 代码
    目录 lxmlwindows安装 读取示例 可视化 生成示例 上面是代码,下面有调用示例 api调用代码,其实只有几行:这个生成代码也很简 ... [详细]
  • Go语言中的高效排序与搜索算法解析
    在探讨Go语言中高效的排序与搜索算法时,本文深入分析了Go语言提供的内置排序功能及其优化策略。通过实例代码,详细讲解了如何利用Go语言的标准库实现快速、高效的排序和搜索操作,为开发者提供了实用的编程指导。 ... [详细]
  • 深入学习 Python 中的 xlrd 模块:掌握 Excel 文件读取技巧
    本文深入探讨了 Python 中的 xlrd 模块,重点介绍了如何高效读取 Excel 文件(包括 xlsx 和 xls 格式)。同时,文章还详细讲解了 xlwt 模块在 Excel 文件写操作中的应用。此外,文中列举了常见单元格数据类型及其处理方法,为读者提供了全面的实践指导。 ... [详细]
  • 在开发Xamarin.Forms应用程序时,遇到了使用Entity Framework Core 3.0访问SQLite数据库时 `Database.MigrateAsync` 方法调用的问题。本文详细探讨了该问题的根源,并提供了一种有效的解决方案,确保数据库迁移能够顺利执行。此外,还介绍了如何配置和优化EF Core以提高应用性能和稳定性。 ... [详细]
  • 本课程详细解析了Spring AOP的核心概念及其增强机制,涵盖前置增强、后置增强和环绕增强等类型。通过具体示例,深入探讨了如何在实际开发中有效运用这些增强技术,以提升代码的模块化和可维护性。此外,还介绍了Spring AOP在异常处理和性能监控等场景中的应用,帮助开发者更好地理解和掌握这一强大工具。 ... [详细]
  • RancherOS 是由 Rancher Labs 开发的一款专为 Docker 设计的轻量级 Linux 发行版,提供了一个全面的 Docker 运行环境。其引导镜像仅 20MB,非常适合在资源受限的环境中部署。本文将详细介绍如何在 ESXi 虚拟化平台上安装和配置 RancherOS,帮助用户快速搭建高效、稳定的容器化应用环境。 ... [详细]
  • 本文深入探讨了 Spring Cloud 微服务架构中 Gateway 组件的应用,详细介绍了其在实现高效请求路由与过滤方面的关键作用。文章首先从基本配置入手,逐步讲解了如何通过静态路由和动态路由实现灵活的服务访问控制。此外,还特别介绍了如何配置 Gateway 以自动从 Nacos 服务注册中心拉取服务列表,进一步提升系统的可维护性和扩展性。 ... [详细]
  • SpringBoot启动脚本详解:BAT文件应用与基础入门指南(SpringBoot系列第1篇)
    如果你还在为SSM框架的复杂搭建过程和繁琐的配置文件而烦恼,那么SpringBoot将是你的一大福音。作为SpringBoot系列的第一篇文章,本文详细介绍了如何使用BAT文件来启动SpringBoot应用,并提供了基础入门指南,帮助开发者快速上手,简化开发流程。 ... [详细]
  • 掌握DSP必备的56个核心问题,我已经将其收藏以备不时之需! ... [详细]
  • 如何判断一个度序列能否构成简单图——哈维尔-哈基米算法的应用与解析 ... [详细]
  • 随着越来越多的应用程序采用JSON格式作为响应数据,基于Spring Framework构建的服务端应用也广泛采用了这一实践。本文将详细介绍如何在Spring 4.x版本的MVC框架中配置和实现HTTP请求返回JSON数据流,涵盖相关配置、依赖管理和代码示例,帮助开发者高效地实现这一功能。 ... [详细]
  • 技术日志:深入探讨Spark Streaming与Spark SQL的融合应用
    技术日志:深入探讨Spark Streaming与Spark SQL的融合应用 ... [详细]
author-avatar
幽咽小香
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有