Golang TCP网络编程

news2024/12/23 13:13:45

文章目录

  • 网络编程介绍
  • TCP网络编程
    • 服务器监听
    • 客户端连接服务器
    • 服务端获取连接
    • 向连接中写入数据
    • 从连接中读取数据
    • 关闭连接/监听器
  • 简易的TCP回声服务器
    • 效果展示
    • 服务端处理逻辑
    • 客户端处理逻辑

网络编程介绍

网络编程介绍

  • 网络编程是指通过计算机网络实现程序间通信的一种编程技术,涉及到在不同计算机之间建立连接、传输数据和协议解析等操作。
  • 套接字(Socket)编程是网络编程的一种实现方式,其提供了一种机制,使得应用程序能够通过网络进行数据传输和通信。
  • Go中的net包是标准库中提供的网络编程包,是基于套接字编程的一种实现方式,提供了对TCP、UDP、IP、ICMP、Unix域套接字等常见网络协议的支持,通过net包可以完成创建套接字、建立连接、发送和接收数据等操作,实现网络通信。

TCP网络编程

服务器监听

服务器监听

在Go的net包中,Listen函数用于创建并返回一个网络监听器(Listener),以监听指定网络地址和端口上的连接请求。该函数的函数原型如下:

func Listen(network, address string) (Listener, error)

参数说明:

  • network:用于指定网络类型,其值必须是"tcp", “tcp4”, “tcp6”, “unix"或"unixpacket”。
  • address:用于指定需要被监听的IP地址和端口号,格式为"host:port"。

返回值说明:

  • 第一个返回值:表示创建的网络监听器。
  • 第二个返回值:如果创建网络监听器过程中出错,将返回非nil的错误值。

通过Listen函数创建得到的网络监听器是Listener类型的,该类型是一个接口类型,其定义如下:

type Listener interface {
	// Accept waits for and returns the next connection to the listener.
	Accept() (Conn, error)

	// Close closes the listener.
	// Any blocked Accept operations will be unblocked and return errors.
	Close() error

	// Addr returns the listener's network address.
	Addr() Addr
}

Listener接口中各方法说明:

  • Accept方法:从底层获取下一个已经建立好的连接给监听器。
  • Close方法:关闭监听器。
  • Addr方法:返回监听器对应的网络地址(由IP地址和端口号组成)。

当使用Listen函数创建TCP类型的监听器时,其返回的监听器底层具体的类型是TCPListener,其定义如下:

type TCPListener struct {
	fd *netFD
	lc ListenConfig
}

TCPListener结构体各字段说明:

  • fd:对底层网络文件描述符的封装,提供了对网络连接的读写和控制操作。
  • lc:用于配置监听器创建的行为,比如设置监听地址、控制网络参数等。

TCPListener结构体中的fd字段是netFD类型的,其定义如下:

type netFD struct {
	pfd poll.FD

	// immutable until Close
	family      int
	sotype      int
	isConnected bool // handshake completed or use of association with peer
	net         string
	laddr       Addr
	raddr       Addr
}

netFD结构体各字段说明:

  • pfd:用于与底层的操作系统文件描述符进行交互。
  • family:表示套接字的协议家族,比如IPv4或IPv6。
  • sotype:表示套接字的类型,比如TCP或UDP
  • isConnected:表示连接是否已完成握手或与对方建立关联。
  • net:表示网络协议,比如"tcp"或"udp"。
  • laddr:表示本地网络连接的地址。
  • raddr:表示远程网络连接的地址。

服务器监听示例

服务器监听示例如下:

package main

import (
	"fmt"
	"net"
)

func main() {
	// 服务器监听
	listen, err := net.Listen("tcp", "0.0.0.0:8081")
	if err != nil {
		fmt.Printf("listen error, err = %v\n", err)
		return
	}
	defer listen.Close()
	fmt.Println("listen success...")
}

说明一下:

  • 服务器在创建监听套接字时,将其绑定到0.0.0.0(通常表示为INADDR_ANY)地址,这样服务器就可以同时监听和接受来自不同网络接口的连接请求,而不需要为每个接口分别创建监听套接字。
  • 为了避免网络文件描述符资源泄露,在创建监听器后及时利用defer机制关闭监听器。监听器被关闭后会停止监听新的连接请求,并且任何被阻塞的Accept操作都会被解除阻塞并返回错误。

客户端连接服务器

客户端连接服务器

在Go的net包中,Dial函数用于客户端应用程序与远程服务器建立连接。该函数的函数原型如下:

func Dial(network, address string) (Conn, error)

参数说明:

  • network:用于指定网络协议,比如"tcp", "udp"等。
  • address:用于指定连接的目标地址。

返回值说明:

  • 第一个返回值:表示建立的网络连接。
  • 第二个返回值:如果建立网络连接过程中出错,将返回非nil的错误值。

通过Dial函数建立得到的连接Conn类型的,该类型是一个接口类型,其定义如下:

type Conn interface {
	// Read reads data from the connection.
	Read(b []byte) (n int, err error)

	// Write writes data to the connection.
	Write(b []byte) (n int, err error)

	// Close closes the connection.
	// Any blocked Read or Write operations will be unblocked and return errors.
	Close() error

	// LocalAddr returns the local network address, if known.
	LocalAddr() Addr

	// RemoteAddr returns the remote network address, if known.
	RemoteAddr() Addr

	// SetDeadline sets the read and write deadlines associated
	// with the connection. It is equivalent to calling both
	// SetReadDeadline and SetWriteDeadline.
	SetDeadline(t time.Time) error

	// SetReadDeadline sets the deadline for future Read calls
	// and any currently-blocked Read call.
	SetReadDeadline(t time.Time) error

	// SetWriteDeadline sets the deadline for future Write calls
	// and any currently-blocked Write call.
	SetWriteDeadline(t time.Time) error
}

Conn接口中各方法说明:

  • Read方法:从连接中读取数据。
  • Write方法:向连接中写入数据。
  • Close方法:关闭连接。
  • LocalAddr方法:返回连接对应的本地网络地址(由IP地址和端口号组成)。
  • RemoteAddr方法:返回连接对应的远程网络地址(由IP地址和端口号组成)。
  • SetDeadline方法:设置连接读取和写入的截止时间。
  • SetReadDeadline方法:设置连接读取的截止时间。
  • SetWriteDeadline方法:设置连接写入的截止时间。

当使用Dial函数与TCP服务器建立连接时,其返回的网络连接底层具体的类型是TCPConn,其定义如下:

type TCPConn struct {
	conn
}

TCPConn结构体中仅嵌套了一个conn类型的匿名结构体,其定义如下:

type conn struct {
	fd *netFD
}

可以看到,conn结构体中的fd字段与TCPListener结构体中的fd字段的类型相同,它们都是对底层网络文件描述符的封装,提供了对网络连接的读写和控制操作。

客户端连接服务器示例

客户端连接服务器示例如下:

package main

import (
	"fmt"
	"net"
)

func main() {
	// 客户端连接服务器
	conn, err := net.Dial("tcp", "127.0.0.1:8081")
	if err != nil {
		fmt.Printf("connect server error, err = %v\n", err)
		return
	}
	defer conn.Close()
	fmt.Println("connect server success...")
}

说明一下:

  • 当客户端连接TCP服务器时,服务器必须处于监听状态。
  • 为了避免网络文件描述符资源泄露,客户端在与服务器建立连接后及时利用defer机制关闭连接。

服务端获取连接

服务端获取连接

在创建TCP网络监听器后,调用Listener接口的Accept方法,本质调用的是TCPListener的Accept方法,该方法用于从底层获取下一个已经建立好的连接给监听器,如果底层没有建立好的连接则会进行阻塞等待。该方法的原型如下:

func (l *TCPListener) Accept() (Conn, error)

返回值说明:

  • 第一个返回值:表示获取到的与客户端建立好的连接。
  • 第二个返回值:如果在获取连接过程中出错,将返回非nil的错误值。

服务端获取连接示例

服务端获取连接示例如下:

package main

import (
	"fmt"
	"net"
)

func process(conn net.Conn) {
	defer conn.Close()

	fmt.Printf("handle a link %v...\n", conn.RemoteAddr())
}

func main() {
	// 服务器监听
	listen, err := net.Listen("tcp", "0.0.0.0:8081")
	if err != nil {
		fmt.Printf("listen error, err = %v\n", err)
		return
	}
	defer listen.Close()
	fmt.Println("listen success...")

	for {
		fmt.Println("waiting client connect...")
		// 服务端获取连接
		conn, err := listen.Accept()
		if err != nil {
			fmt.Printf("accept error, err = %v\n", err)
			continue
		}
		fmt.Printf("get a link from %v...\n", conn.RemoteAddr())
		// 开启新协程为客户端提供服务
		go process(conn)
	}
}

说明一下:

  • 网络监听器的任务就是不断调用Accept方法,从底层获取已经建立好的连接并为其提供服务,通常在获取到一个连接后会开启一个新协程为其提供服务,而主协程则继续调用Accept方法获取新的连接。
  • 为了让新协程能够获取需要被处理的连接,需要将对应的连接通过参数传递的方式,传递给协程对应的处理函数。此外,为了避免网络文件描述符资源泄露,需要在处理函数中利用defer机制关闭连接,保证连接处理完毕后能够及时关闭连接。

向连接中写入数据

向连接中写入数据

在创建TCP连接后,调用Conn接口的Write方法,本质调用的是TCPConn的Write方法,该方法用于向连接中写入数据。该方法的原型如下:

func (c *TCPConn) Write(b []byte) (int, error)

参数说明:

  • b:表示要写入连接的数据。

返回值说明:

  • 第一个返回值:表示实际写入的字节数。
  • 第二个返回值:如果在写入数据过程中出错,将返回非nil的错误值。

从连接中读取数据

从连接中读取数据

在创建TCP连接后,调用Conn接口的Read方法,本质调用的是TCPConn的Read方法,该方法用于从连接中读取数据。该方法的原型如下:

func (c *TCPConn) Read(b []byte) (int, error)

参数说明:

  • b:输出型参数,用于存储读取到的数据。

返回值说明:

  • 第一个返回值:表示实际读取的字节数。
  • 第二个返回值:如果在读取数据过程中出错,将返回非nil的错误值。

关闭连接/监听器

关闭连接/监听器

为了避免网络文件描述符泄露,TCP网络监听器和TCP连接在使用完毕后都需要及时将其关闭,对应调用的分别是TCPListener和TCPConn的Close方法,这两个方法的原型如下:

func (l *TCPListener) Close() error
func (c *TCPConn) Close() error

返回值说明:

  • 如果在关闭监听器或关闭连接过程中出错,将返回非nil的错误值。

简易的TCP回声服务器

效果展示

效果展示

为了演示使用net包实现网络通信,下面实现了一个简易的TCP回声服务器,其功能如下:

  • 服务端能够同时处理多个客户端的连接请求,在为每个客户端提供服务时,能够将各个客户端发来的数据显示在服务端,同时将客户端发来的数据再发回给客户端。
  • 客户端在连接到服务器后,能够不断从控制台读取用户输入的数据发送给服务端,并将服务端发来的数据显示在客户端,用户在控制台输入exit能够退出客户端。

最终效果如下:

在这里插入图片描述

服务端处理逻辑

服务端处理逻辑

服务端处理逻辑如下:

  • 主协程调用Listen函数完成服务器的监听后,通过监听器不断调用Accept方法从底层获取已经建立好的连接,并为每一个获取到的连接创建一个新协程为其提供服务,而主协程则继续获取新的连接。
  • 每个新协程在为其对应的连接提供服务时,通过调用连接的Read方法不断读取客户端发来的数据,将数据其显示在服务端,同时通过调用连接的Write方法将客户端发来的数据再发回给客户端。
  • 每个新协程在为其对应的连接提供服务的过程中,如果从连接中读取数据或向连接中写入数据时出错,或是客户端退出,则通过调用连接的Close方法将对应的连接关闭。

服务端代码如下:

package main

import (
	"fmt"
	"io"
	"net"
)

func process(conn net.Conn) {
	defer conn.Close()

	data := make([]byte, 1024)
	for {
		// 1、读取客户端发来的数据
		n, err := conn.Read(data)
		if err != nil {
			if err == io.EOF {
				fmt.Printf("client %v quit\n", conn.RemoteAddr())
			} else {
				fmt.Printf("read client message error, err = %v\n", err)
			}
			return
		}
		fmt.Printf("client message[%v]: %v\n", conn.RemoteAddr(), string(data[:n]))

		// 2、发送数据给客户端
		len, err := conn.Write(data[:n])
		if err != nil || len != n {
			fmt.Printf("send back message error, err = %v\n", err)
			return
		}
	}
}

func main() {
	// 服务器监听
	listen, err := net.Listen("tcp", "0.0.0.0:8081")
	if err != nil {
		fmt.Printf("listen error, err = %v\n", err)
		return
	}
	defer listen.Close()
	fmt.Println("listen success...")

	for {
		fmt.Println("waiting client connect...")
		// 服务端获取连接
		conn, err := listen.Accept()
		if err != nil {
			fmt.Printf("accept error, err = %v\n", err)
			continue
		}
		fmt.Printf("get a link from %v...\n", conn.RemoteAddr())
		// 开启新协程为客户端提供服务
		go process(conn)
	}
}

说明一下:

  • 当服务端从某个连接中读取数据时,如果该连接对应的客户端已经将连接关闭,那么服务端的读操作将会返回io.EOF错误。

客户端处理逻辑

客户端处理逻辑

客户端处理逻辑如下:

  • 客户端在调用Dial函数与服务端建立连接后,不断读取用户的输入,通过调用连接Write方法将用户输入的数据发送给服务端,然后通过调用连接的Read方法读取服务端发来的数据并显示在客户端,如果用户输入exit则调用连接的Close方法将连接关闭。

客户端代码如下:

package main

import (
	"bufio"
	"fmt"
	"net"
	"os"
)

func main() {
	// 客户端连接服务器
	conn, err := net.Dial("tcp", "127.0.0.1:8081")
	if err != nil {
		fmt.Printf("connect server error, err = %v\n", err)
		return
	}
	defer conn.Close()
	fmt.Println("connect server success...")

	reader := bufio.NewReader(os.Stdin)
	data := make([]byte, 1024)
	for {
		// 1、读取用户输入
		str, err := reader.ReadString('\n')
		if err != nil {
			fmt.Printf("read input error, err = %v\n", err)
			continue
		}
		str = str[:len(str)-2] // 去掉\r\n
		if str == "exit" {
			fmt.Printf("exit success...")
			break
		}

		// 2、发送数据给服务端
		n, err := conn.Write([]byte(str))
		if err != nil || n != len(str) {
			fmt.Printf("send message error, err = %v\n", err)
			continue
		}
		fmt.Printf("send %d byte message to server...\n", n)

		// 3、读取服务端发来的数据
		n, err = conn.Read(data)
		if err != nil {
			fmt.Printf("read message error, err = %v\n", err)
			continue
		}
		fmt.Printf("server message: %v\n", string(data[:n]))
	}
}

说明一下:

  • Stdin、Stdout和Stderr是os包中的全局变量,分别表示标准输入流、标准输出流和标准错误流。
  • 客户端在读取用户输入时,通过bufio包中的Reader,以带缓冲的方式每次从标准输入流中读取一行数据。
  • Windows系统中通常使用\r\n作为换行符,因此客户端在每次读取一行用户输入的数据后需要将末尾的两个字符去掉。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1796583.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SLC Flash SD芯片:高性能存储的优选

SLC Flash SD芯片是一种采用单阶存储单元(SingleLevel Cell,SLC)技术的Secure Digital(SD)存储卡。SLC技术以其快速的传输速度、低功耗和较长的存储单元寿命而闻名。 MK米客方德 SLC Flash的优势 1. 快速的传输速度&a…

蓝牙AOA基站定位的优势与应用前景

在科技飞速发展的今天,人们对于定位技术的精度和稳定性要求越来越高。蓝牙AOA基站定位技术应运而生,以其高精度和低通信开销的特点,正逐渐改变我们对室内定位的认知。本文我们就一起来具体了解一下关于蓝牙AOA基站定位技术的优势及应用前景&a…

实现飞书机器人推送消息到指定群组或者用户

实现飞书机器人推送消息到指定群组或者用户 1 简介2 创建飞书应用2.1 注册登录2.2 创建应用2.3 添加应用能力2.4 权限管理3 发布应用4 代码示例4.1 获取应用ID与token4.2 使用Python SDK4.3 简单示例4.4 获取用户或机器人所在的群列表4.5 通过手机号或邮箱获取用户 ID4.6 给群组…

【Vue】普通组件的注册使用-局部注册

文章目录 一、组件注册的两种方式二、使用步骤三、练习 一、组件注册的两种方式 局部注册:只能在注册的组件内使用 ① 创建 .vue 文件 (三个组成部分) 以.vue结尾的组件,一般也叫做 单文件组件,即一个组件就是组件里的全部内容 ② 在使用的组…

【栈】1096. 花括号展开 II

本文涉及知识点 栈 LeetCode 1096. 花括号展开 II 如果你熟悉 Shell 编程,那么一定了解过花括号展开,它可以用来生成任意字符串。 花括号展开的表达式可以看作一个由 花括号、逗号 和 小写英文字母 组成的字符串,定义下面几条语法规则&…

树莓派4B_OpenCv学习笔记4:测试摄像头_imread加载显示图像_imwrite保存图片

今日继续学习树莓派4B 4G:(Raspberry Pi,简称RPi或RasPi) 本人所用树莓派4B 装载的系统与版本如下: 版本可用命令 (lsb_release -a) 查询: Opencv 版本是4.5.1: 今日对之前的测试CSI摄像头函数进行一些理解说明&#x…

混合关键性系统技术【同构异构】【SMP、AMP、BMP】【嵌入式虚拟化】

混合关键性系统技术【同构异构】【SMP、AMP、BMP】【嵌入式虚拟化】 1 介绍1.1 概述openEuler Embedded 的运行模式openEuler Embedded 混合关键性系统技术架构UniProton 1.2 同构异构区别 【硬件侧】1.3 系统架构【SMP、AMP、BMP】多核处理器平台的系统架构 【软件侧】【SMP、…

【RuoYi】框架中使用wangdietor富文本编辑器

一、前言 在上篇博客中,介绍了RuoYi中如何实现文件的上传与下载,那么这篇博客讲一下如何在RuoYi中使用富文本编辑器,这部分的内容是向B站程序员青戈学习的,当然我这里就会把学到的内容做一个总结,当然也会说…

Java基础语法---集合---ArrayList

ArrayList是什么 ArrayList可以看作是一个动态数组,提供了自动扩容的能力,意味着它能够根据需要自动调整其大小以容纳更多的元素,而无需预先指定数组的容量。 使用ArrayList需要加入包 import java.util.ArryList ArrayList与普通数组的不同…

【C++】─篇文章带你熟练掌握 map 与 set 的使用

目录 一、关联式容器二、键值对三、pair3.1 pair的常用接口说明3.1.1 [无参构造函数](https://legacy.cplusplus.com/reference/utility/pair/pair/)3.1.2 [有参构造函数 / 拷贝构造函数](https://legacy.cplusplus.com/reference/utility/pair/pair/)3.1.3 [有参构造函数](htt…

bootstrap5-学习笔记1-容器+布局+按钮+工具

参考: Bootstrap5 教程 | 菜鸟教程 https://www.runoob.com/bootstrap5/bootstrap5-tutorial.html Spacing Bootstrap v5 中文文档 v5.3 | Bootstrap 中文网 https://v5.bootcss.com/docs/utilities/spacing/ 之前用bootstrap2和3比较多,最近用到了5&a…

【FAS】《Liveness Detection on Face Anti-spoofing》

文章目录 原文总结与评价CNN-RNN vs 三维卷积作者的方法 原文 [1]欧阳文汉.反人脸图像欺诈的活体识别方法研究[D].浙江大学,2020.DOI:10.27461/d.cnki.gzjdx.2020.002675. 总结与评价 时序运动信息与传统的空间纹理信息相结合 基于相位平移的运动放大算法不错 视觉大小细胞…

每日一题《leetcode--LCR 029.循环有序列表的插入》

https://leetcode.cn/problems/4ueAj6/ 这道题整体上想插入数据有三种情况: 1、整个列表是空列表,需要返回插入的结点 2、整个列表只有一个结点,需要在头结点后插入新结点,随机把新结点的next指向头结点 3、整个列表的结点 >1 …

关于飞速(FS)800G光模块的技术问答

随着云计算、物联网(IoT)和大数据等技术的快速发展,对网络带宽和传输速率的需求越来越大。飞速(FS)800G光模块的引入旨在满足对高速数据传输的需求,该光模块采用先进的调制解调技术和高密度光电子元件&…

STM32F103VE和STM32F407VE的引脚布局

STM32F103VE vs STM32F407VE 引脚对比表 引脚 STM32F103VE STM32F407VE 备注 1 VSS VSS 地 2 VDD VDD 电源 3 VSSA VSSA 模拟地 4 VDDA VDDA 模拟电源 5 OSC_IN OSC_IN 外部时钟输入 6 OSC_OUT OSC_OUT 外部时钟输出 7 NRST NRST 复位 8 PC13 (GPIO) PC13 (GPIO) GPIO 9 PC14 (…

鸿蒙全栈开发-一文读懂鸿蒙同模块不同模块下的UIAbility跳转详解

前言 根据第三方机构Counterpoint数据,截至2023年三季度末,HarmonyOS在中国智能手机操作系统的市场份额已经提升至13%。短短四年的时间,HarmonyOS就成长为仅次于安卓、苹果iOS的全球第三大操作系统。 因此,对于鸿蒙生态建设而言&a…

依赖注入方式和自动加载原理

依赖注入 Spring提供了依赖注入的功能,方便我们管理和使用各种Bean,常见的方式有: 字段注入(Autowired 或 Resource)构造函数注入set方法注入 在以往代码中,我们经常利用Spring提供的Autowired注解来实现…

C语言操作UTF-8编码和GBK编码的文件的示例

一、基本介绍 在C语言中,处理文件编码(如UTF-8或GBK)时,需要注意C标准库中的文件操作函数(如fopen, fread, fwrite, fclose等)并不直接支持Unicode或特定字符集的编码。这些函数通常处理字节流&#xff0c…

FreeRTOS基础(十一):消息队列

本文将详细全方位的讲解FreeRTOS的消息队列,其实在FreeRTOS中消息队列的重要性也不言而喻,与FreeRTOS任务调度同等重要,因为后面的各种信号量基本都是基于消息队列的。 目录 一、消息队列的简介 1.1 产生的原因 1.2 消息队列的解决办法 …

防病毒克星——白名单可信系统

白名单作为一种网络安全措施,其核心概念在于限制用户只能访问网络所有者定义的受信任内容。这种机制在保护系统免受恶意软件、病毒等攻击方面发挥着重要作用。然而,关于白名单是否可以防病毒的问题,实际上涉及了多个方面的考虑。 首先&#x…