golang:TCP总结
在TCP/IP协议中,“IP地址+TCP或UDP端口号”唯一标识网络通讯中的一个进程。“IP地址+端口号”就对应一个socket。欲建立连接的两个进程各自有一个socket来标识,那么这两个socket组成的socket pair就唯一标识一个连接。因此可以用Socket来描述网络连接的一对一关系。
常用的Socket类型有两种:流式Socket(SOCK_STREAM)和数据报式Socket(SOCK_DGRAM)。流式是一种面向连接的Socket,针对于面向连接的TCP服务应用;数据报式Socket是一种无连接的Socket,对应于无连接的UDP服务应用。
套接字通讯原理示意

TCP的C/S架构

在整个通信过程中,服务器端有两个socket参与进来,但用于通信的只有conn这个socket。它是由 listener创建的。隶属于服务器端。客户端有一个socket参与进来。
net.Listen() 建立一个用于连接监听的套接字
listen.Accept() // 阻塞监听客户端连接请求,成功用于连接,返回用于通信的socket
net.Dial() 客户端向服务端发起连接建立一个socket连接

并发的C/S模型通信
Server
Accept()函数的作用是等待客户端的链接,如果客户端没有链接,该方法会阻塞。如果有客户端链接,那么该方法返回一个Socket负责与客户端进行通信。所以,每来一个客户端,该方法就应该返回一个Socket与其通信,因此,可以使用一个死循环,将Accept()调用过程包裹起来。
需要注意,实现并发处理多个客户端数据的服务器,就需要针对每一个客户端连接,单独产生一个Socket,并创建一个单独的goroutine与之完成通信。
package main
import (
"fmt"
"net"
"strings"
)
func handleConnect(conn net.Conn){
var (
b []byte
err error
n int
)
fmt.Println(conn.RemoteAddr(),"建立连接.")
defer conn.Close()
b = make([]byte,4096)
// 客户端可能持续不断的发送数据,因此接收数据的过程可以放在for循环中,服务端也持续不断的向客户端返回处理后的数据。
for {
n,err = conn.Read(b)
content := strings.Trim(string(b[:n]),"\r\n") // window中传送的内容存在换行符,作为判断时需要删除
// 当客户端退出,服务端从chan中读取内容时是没有的,因此的到0 或者客户端主动退出输入exit或者quit
if n == 0 || content == "exit" || content == "quit" {
fmt.Println("客户端退出:",conn.RemoteAddr())
return
}
if err != nil {
fmt.Println(err)
return
}
if _,err = conn.Write([]byte(fmt.Sprintf("server reply:%s",b[:n])));err !=nil {
fmt.Println(err)
return
}
fmt.Println("client send: ",content)
}
}
func main() {
var (
listener net.Listener
err error
conn net.Conn
)
// 建立一个用于连接监听的套接字
if listener, err = net.Listen("tcp", "10.0.0.1:8088"); err != nil {
fmt.Println(err)
return
}
defer listener.Close()
fmt.Println("waiting client connect.")
// 阻塞监听客户端连接请求,成功用于连接,返回用于通信的socket
for {
if conn, err = listener.Accept(); err != nil {
fmt.Println(err)
return
}
go handleConnect(conn)
}
}
使用nc作为客户端向服务端发送信息

自定义客户端
客户端需要持续的向服务端发送数据,同时也要接收从服务端返回的数据。因此可将发送和接收放到不同的协程中。
- 主协程循环接收服务器回发的数据(该数据应已转换为大写),并打印至屏幕;
- 子协程循环从键盘读取用户输入数据。
- 读取键盘输入可使用
os.Stdin.Read()。
注意事项:
- 服务端有对 exit返回的是
io.EOF - 当服务端断开时,chan读取的信息就为0了即服务端已经退出,如果客户端不退出会一直报错
package main
import (
"fmt"
"io"
"net"
"os"
"strings"
)
func main() {
var (
conn net.Conn
err error
n int
)
if conn, err = net.Dial("tcp", "10.0.0.1:8088"); err != nil {
fmt.Println(err, 111)
return
}
defer conn.Close()
go func() {
str := make([]byte, 1024)
for {
n, err := os.Stdin.Read(str)
content := strings.ToLower(strings.Trim(string(str[:n]), "\r\n"))
if n == 0 {
fmt.Println("与服务端断开连接")
return
}
if err == io.EOF || content == "quit" {
return
}
if err != nil {
fmt.Println(1, err)
continue
}
_, err = conn.Write([]byte(content))
if err != nil {
fmt.Println(111, err)
return
}
}
}()
byt := make([]byte, 1024)
for {
if _, err = conn.Read(byt); err != nil {
if err == io.EOF {
return
}
fmt.Println(err)
continue
}
fmt.Println("server reply:", string(byt[:n]))
}
}

golang:TCP总结的更多相关文章
- Golang TCP转发到指定地址
Golang TCP转发到指定地址 第二个版本,设置指定ip地址 代码 // tcpForward package main import ( "fmt" "net&qu ...
- Golang tcp转发 remoteAddr错误
Golang tcp 转发 第一版本 accept获取的Conn里的localAddr做为源地址,remoteAddr来做为目的地址 // tcpForward package main import ...
- 6行代码解决golang TCP粘包
转自:https://studygolang.com/articles/12483 什么是TCP粘包问题以及为什么会产生TCP粘包,本文不加讨论.本文使用golang的bufio.Scanner来实现 ...
- Golang Tcp粘包处理(转)
在用golang开发人工客服系统的时候碰到了粘包问题,那么什么是粘包呢?例如我们和客户端约定数据交互格式是一个json格式的字符串: {"Id":1,"Name" ...
- golang tcp keepalive实践
前文中已经介绍了TCP keep alive的做了详尽说明,本文结合golang,介绍如何使用TCP keep alive. 目前golang net包不提供TCP keep alive 空闲多长时间 ...
- 【GoLang】golang TCP 粘包处理 示例
参考资料: http://www.01happy.com/golang-tcp-socket-adhere/
- <转>Go语言TCP Socket编程
授权转载: Tony Bai 原文连接: https://tonybai.com/2015/11/17/tcp-programming-in-golang/ Golang的主要 设计目标之一就是面向大 ...
- Go语言学习之9 网络协议TCP、Redis与聊天室
主要内容 1. Tcp编程2. redis使用 1. Tcp编程 (1)简介 Golang是谷歌设计开发的语言,在Golang的设计之初就把高并发的性能作为Golang的主要特性之一,也是 ...
- Go语言TCP Socket编程
Golang的主要 设计目标之一就是面向大规模后端服务程序,网络通信这块是服务端 程序必不可少也是至关重要的一部分.在日常应用中,我们也可以看到Go中的net以及其subdirectories下的 ...
- go socket
https://tonybai.com/2015/11/17/tcp-programming-in-golang/ Golang的主要 设计目标之一就是面向大规模后端服务程序,网络通信这块是服务端 程 ...
随机推荐
- Prometheus联邦
联邦使得一个 Prometheus 服务器可以从另一个 Prometheus 服务器提取选定的时序. 1. 使用场景 Prometheus 联邦有不同的使用场景.通常,联邦被用来实现可扩展的 Prom ...
- 第13 章 : Kubernetes 网络概念及策略控制
Kubernetes 网络概念及策略控制 本文将主要分享以下 5 方面的内容: Kubernetes 基本网络模型: Netns 探秘: 主流网络方案简介: Network Policy 的用处: 思 ...
- Rancher 安装和使用-实践
Rancher 安装和使用 Rancher是一个完整的,开源的平台,用于在生产环境中部署和管理容器.它包括Kubernetes,Mesos和Docker Swarm的商业支持发行版,使得在任何基础架构 ...
- 【洛谷】P1294 高手去散步
题目背景 高手最近谈恋爱了.不过是单相思."即使是单相思,也是完整的爱情",高手从未放弃对它的追求.今天,这个阳光明媚的早晨,太阳从西边缓缓升起.于是它找到高手,希望在晨读开始之前 ...
- Python基础(二十):面向对象“类”第三课——类成员
知识点: 类属性与实例属性: 类方法与实例方法: 静态方法: 类属性与实例属性 类属性与实例属性的区别 属性的绑定不同 类属性与当前类相关(绑定的是当前类),与当前类创建的任何对象无关: 实例属性与当 ...
- 201871010203-陈鹏昱 实验二 个人项目—《D{0-1}KP问题》项目报告
项目 内容 课程班级博客链接 班级博客 这个作业要求链接 作业要求 我的课程学习目标 运用科学高效的方法学习软件工程的理论和知识 这个作业在哪些方面帮助我实现学习目标 掌握软件项目个人开发流程,掌握G ...
- MySQL数据库高级一:架构介绍
两天半就可以 严禁使用 精通 在简历上 了解的越多,越比他人有优势 linux的mysql需要使用中文字符集那么就要修改配置文件 1.mysql的linux版 安装和卸载不说了 2.逻辑架构 总体概况 ...
- HarmonyOS开发者看过来,HDD上海站传递的重要信息都在这里
4月17日,颇有HarmonyOS年度总结性质的HarmonyOS开发者日活动上海站正式开始. 活动中,华为消费者业务AI与智慧全场景业务部副总裁段孟对HarmonyOS生态建设的最新进展做了发言,并 ...
- git推送代码报错:fatal: The current branch master has no upstream branch. To push the current branch and set the remote as upstream
情景再现 远程新建仓库,然后本地 git bash执行以下代码 git init git add . git commit -m 'xxx' git remote add origin https:/ ...
- Arch! 从安装开始
Arch! 从安装开始 事实上Arch的安装Arch Wiki Installation Guide,已经非常详细了 但是初次面对这些东西时肯定非常迷茫,根本不知道这些东西是在干嘛?为什么要这么干? ...