Golang TCP网络编程

文章目录

  • 网络编程介绍
  • 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 *netFDlc ListenConfig
}

TCPListener结构体各字段说明:

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

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

type netFD struct {pfd poll.FD// immutable until Closefamily      intsotype      intisConnected bool // handshake completed or use of association with peernet         stringladdr       Addrraddr       Addr
}

netFD结构体各字段说明:

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

服务器监听示例

服务器监听示例如下:

package mainimport ("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 mainimport ("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 mainimport ("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 mainimport ("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 mainimport ("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\nif 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.mzph.cn/pingmian/21829.shtml

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

相关文章

三十五、openlayers官网示例Dynamic Data——在地图上加载动态数据形成动画效果

官网demo地址: Dynamic Data 初始化地图 const tileLayer new TileLayer({source: new OSM(),});const map new Map({layers: [tileLayer],target: "map",view: new View({center: [0, 0],zoom: 2,}),}); 创建了三个样式 const imageStyle new Style(…

黑马微服务实用篇知识梳理

1、微服务治理 1.1服务注册与发现Eureka和Nacos a、nacos和eureka,二者都支持服务注册与发现,但nacos还包括了动态配置管理、服务健康监测、动态路由等功能,是更全面的服务管理平台 b、eureka需要独立部署为服务并运行,需要自行搭…

师彼长技以助己(3)逻辑思维

师彼长技以助己(3)逻辑思维 前言 上一篇文章进行了工程思维和产品思维的测试,并介绍了几个比较重要的产品思维模型。接下来本篇介绍工程思维。(注意产品思维并不代表产品经理思维,工程思维也并不代表工程师思维&…

【用户画像】用户偏好购物模型BP

一、前言 用户购物偏好模型BP(Buyer Preferences Model)旨在通过对用户购物行为的深入分析和建模,以量化用户对不同商品或服务的偏好程度。该模型对于电商平台、零售商以及其他涉及消费者决策的商业实体来说,具有重要的应用价值。…

linux,lseek,append用法

打开写的.c文件 内容为 代码 <sys/stat.h> #include <fcntl.h> #include<stdio.h> #include<unistd.h> #include<string.h>//off_t lseek(int fd, off_t offset, int whence); //int open(const char *pathname, int flags); //int open(const …

二类电商想做爆品还是得会选品

对于二类电商&#xff0c;尤其是电商新手而言&#xff0c;往往很多人都不知道自己该如何选品&#xff0c;在哪里选品&#xff0c;以及如何谈品等等的难题。想要挑选具有潜力的商品&#xff0c;首先我们需要梳理好商品选择的逻辑&#xff1a; 二类电商的特点&#xff1f;你要投…

装机必备——360驱动大师安装教程

装机必备——360驱动大师安装教程 软件下载 软件名称&#xff1a;360驱动大师2.0beta 软件语言&#xff1a;简体中文 软件大小&#xff1a;13.87M系统要求&#xff1a;Windows7或更高&#xff0c; 32/64位操作系统 硬件要求&#xff1a;CPU2GHz &#xff0c;RAM2G或更高 下载…

36【Aseprite 作图】蒸笼盖——拆解

1 蒸笼盖框架 里圈和外圈的形状都是一样的 扶手处&#xff0c;2 1 2 2 2&#xff08;最好都是2&#xff0c;拐角处用1&#xff09; 2 上色 中间的波浪&#xff0c;是2 2 2 上&#xff08;再 2 2 2 下&#xff09; 下方阴影&#xff0c;左边的阴影&#xff0c;右边的阴影颜色…

冯喜运:6.3黄金原油晚间最新行情及独家操作策略指导

【黄金消息面分析】&#xff1a;在全球经济的波动和不确定性中&#xff0c;黄金作为传统的避险资产&#xff0c;其价格走势和市场分析一直是投资者关注的焦点。本周一&#xff08;北京时间6月3日&#xff09;&#xff0c;现货黄金价格基本持平&#xff0c;交易商正在等待本周公…

一个AI板卡电脑--香橙派 AIpro

本文算是一个开箱测评&#xff0c;主要评估它和一个电脑的距离。 香橙派官网&#xff1a;香橙派(Orange Pi)-Orange Pi官网-香橙派开发板,开源硬件,开源软件,开源芯片,电脑键盘香橙派&#xff08;Orange Pi&#xff09;是深圳市迅龙软件有限公司旗下开源产品品牌;香橙派&#x…

抖音小店怎么找厂家代发?从沟通到发货,全流程不容错过!

哈喽~ 我是电商月月 新手做抖音小店&#xff0c;无货源模式的商家不知道怎么找货源&#xff1f; 今天月月就给大家讲解一下抖音小店从找厂家&#xff0c;到和厂家沟通&#xff0c;最后协商发货的方法步骤都有哪些&#xff1f; 满满干货&#xff0c;不容错过&#xff0c;建议…

游戏找不到d3dcompiler43.dll怎么办,分享5种有效的解决方法

在计算机使用过程中&#xff0c;我们经常会遇到一些错误提示&#xff0c;其中之一就是找不到某个文件。其中&#xff0c;找不到d3dcompiler43.dll是一个常见的问题。这个问题通常出现在运行某些游戏或应用程序时&#xff0c;由于缺少了d3dcompiler43.dll文件&#xff0c;导致程…

openeuler欧拉系统连不上网,ping百度报错,ping: www.baidu.com: Name or service not known

一、现象 使用华为 openeuler 系统连不上网&#xff0c;ping 百度报如下错误 ↓ ping: www.baidu.com: Name or service not known二、原因 没有配置dns服务器 三、解决办法 进入网络配置文件存放文件夹 cd /etc/sysconfig/network-scripts/查看对应的网口文件 ls ps: 不同系…

C语言:(动态内存管理)

目录 动态内存有什么用呢 malloc函数 开辟失败示范 free函数 calloc函数 realloc函数 当然realooc也可以开辟空间 常⻅的动态内存的错误 对NULL指针的解引⽤操作 对动态内存开辟的空间越界访问 对⾮动态开辟内存使⽤free释放 使⽤free释放⼀块动态开辟内存的⼀部分 …

TCP三次握手四次挥手,滑动窗口

TCP三次握手 TCP&#xff08;传输控制协议&#xff09;是一种重要的网络协议&#xff0c;它属于互联网协议套件中的传输层&#xff0c;主要用于在不可靠的互联网上提供可靠的、有序的和无差错的数据传输。下面详细介绍TCP的工作原理&#xff0c;包括其连接建立的三次握手过程。…

类和对象(下)【初始化列表】【static成员】【友元】等..... .及【练习题】

类和对象&#xff08;下&#xff09; 1.再谈构造函数 1.1构造函数体赋值 在创建对象时&#xff0c;编译器通过调用构造函数&#xff0c;给对象中各个成员变量一个合适的初始值。 // 初始化列表 # include<iostream> using namespace std;class Date { public:// 构造…

机器学习第十一次课

前言 从现在开始进入神经网络的领域了 正文 先是一段历史介绍,这个就跳过吧,我觉得这里最重要的就是反向传播这里 反向传播 反向传播&#xff08;Backpropagation&#xff09;是一种训练人工神经网络的算法&#xff0c;它通过计算损失函数关于网络参数的梯度来调整网络参数…

java基础篇(1)

JDK是什么?有哪些内容组成?JDK是Java开发工具包 JVM虚拟机: Java程序运行的地方 核心类库: Java已经写好的东西&#xff0c;我们可以直接用开发工具: javac、java、jdb、jhat.. JRE是什么?有哪些内容组成? JRE是Java运行环境 JVM、核心类库、运行工具 JDK&#xff0c;JRE&…

Vulnhub项目:HACKSUDO: THOR

1、靶机地址 hacksudo: Thor ~ VulnHubhacksudo: Thor, made by Vishal Waghmare. Download & walkthrough links are available.https://vulnhub.com/entry/hacksudo-thor,733/ 2、渗透过程 来了来了&#xff0c;开搞&#xff0c;先看目标 56.161&#xff0c;本机 56.1…

WebGL开发电力数字孪生

WebGL 开发电力数字孪生是一项复杂但极具潜力的任务。电力数字孪生通过创建电力系统的虚拟模型&#xff0c;可以实时监控、分析和优化电力系统的运行状态。以下是开发电力数字孪生的详细步骤。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎交流…