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的主要 设计目标之一就是面向大规模后端服务程序,网络通信这块是服务端 程 ...
随机推荐
- FFMPEG编译问题记录
一.ffmpeg下载与配置 下载地址 FFmpeg/FFmpeg (https://github.com/FFmpeg/FFmpeg) ~$ git clone https://github.com/ ...
- 阿里云 RTC QoS 弱网对抗之变分辨率编码
本文为 QoS 弱网优化系列的第二篇 作者|安基程.田伟峰 审校| 泰一 视频编码中的变分辨率问题及解决 变分辨率在弱网场景的实际应用中非常常见,网络状况不好的时候降低分辨率可以降低码率,减少块效应, ...
- 关于在forEach中使用await的问题
先说需求,根据数组中的ID值,对每个ID发送请求,获取数据进行操作. 首先肯定考虑用forEach 或者 map对数组进行遍历,然后根据值进行操作,但是请求是个异步操作,forEach又是一个同步操作 ...
- Node.js核心入门
前言: 因为以前学习Node.js并没有真正意义上的去学习它,而是粗略的学习了npm的常用命令和Node.js一些模块化的语法,因此昨天花了一天的时间看了<Node.js开发指南>一书.通 ...
- (一)Docker-in-Docker on Kubernetes
1. 场景 请参考docker in docker 文章 2. DinD 我们将采用主机Docker守护程序作为外部守护程序,Docker守护程序作为内部守护程序在容器内运行.运行DinD的一个重要方 ...
- 11. Grub 介绍
Grub 全称:Grand Unified Bootloader grub引导也分为两个阶段stage1阶段和stage2阶段(有些较新的grub又定义了stage1.5阶段). 一般配置文件:/bo ...
- Salesforce学习之路(七)Visualforce结合Reports展示图表
Salesforce作为一款CRM系统,个人觉得最重要的环境便是在于数据的展示和联动,而Salesforce也本身提供了相当强大的功能,Report在展示图表的方面十分强大,前段时间更是宣布以157亿 ...
- Day16_88_通过反射机制执行方法
通过反射机制执行方法 * method.invoke(object,"admin","123"); * 代码 import java.lang.reflect. ...
- Laravel打印sql日志
直接打印 use Log; use DB; DB::connection()->enableQueryLog(); Log::info(DB::getQueryLog()); //print_r ...
- POJ 2396 构造矩阵(上下流)
题意: 要求构造一个矩阵,给你行和,列和,还有一些点的上下范围,输出一个满足题意的矩阵. 思路: 这个题目很经典,这是自己看上下流后接触的第一道题,感觉很基础的一道题目,现在我 ...