在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总结的更多相关文章

  1. Golang TCP转发到指定地址

    Golang TCP转发到指定地址 第二个版本,设置指定ip地址 代码 // tcpForward package main import ( "fmt" "net&qu ...

  2. Golang tcp转发 remoteAddr错误

    Golang tcp 转发 第一版本 accept获取的Conn里的localAddr做为源地址,remoteAddr来做为目的地址 // tcpForward package main import ...

  3. 6行代码解决golang TCP粘包

    转自:https://studygolang.com/articles/12483 什么是TCP粘包问题以及为什么会产生TCP粘包,本文不加讨论.本文使用golang的bufio.Scanner来实现 ...

  4. Golang Tcp粘包处理(转)

    在用golang开发人工客服系统的时候碰到了粘包问题,那么什么是粘包呢?例如我们和客户端约定数据交互格式是一个json格式的字符串: {"Id":1,"Name" ...

  5. golang tcp keepalive实践

    前文中已经介绍了TCP keep alive的做了详尽说明,本文结合golang,介绍如何使用TCP keep alive. 目前golang net包不提供TCP keep alive 空闲多长时间 ...

  6. 【GoLang】golang TCP 粘包处理 示例

    参考资料: http://www.01happy.com/golang-tcp-socket-adhere/

  7. <转>Go语言TCP Socket编程

    授权转载: Tony Bai 原文连接: https://tonybai.com/2015/11/17/tcp-programming-in-golang/ Golang的主要 设计目标之一就是面向大 ...

  8. Go语言学习之9 网络协议TCP、Redis与聊天室

    主要内容 1. Tcp编程2. redis使用 1. Tcp编程 (1)简介       Golang是谷歌设计开发的语言,在Golang的设计之初就把高并发的性能作为Golang的主要特性之一,也是 ...

  9. Go语言TCP Socket编程

      Golang的主要 设计目标之一就是面向大规模后端服务程序,网络通信这块是服务端 程序必不可少也是至关重要的一部分.在日常应用中,我们也可以看到Go中的net以及其subdirectories下的 ...

  10. go socket

    https://tonybai.com/2015/11/17/tcp-programming-in-golang/ Golang的主要 设计目标之一就是面向大规模后端服务程序,网络通信这块是服务端 程 ...

随机推荐

  1. (五)SpringBoot启动过程的分析-刷新ApplicationContext

    -- 以下内容均基于2.1.8.RELEASE版本 紧接着上一篇[(四)SpringBoot启动过程的分析-预处理ApplicationContext] (https://www.cnblogs.co ...

  2. ( ) 与 { } 差在哪?-- Shell十三问<第七问>

    ( ) 与 { } 差在哪?-- Shell十三问<第七问> 先说一下,为何要用 ( ) 或 { } 好了. 许多时候,我们在 shell 操作上,需要在一定条件下一次执行多个命令,也就是 ...

  3. 「HTML+CSS」--自定义加载动画【016】

    前言 Hello!小伙伴! 首先非常感谢您阅读海轰的文章,倘若文中有错误的地方,欢迎您指出- 哈哈 自我介绍一下 昵称:海轰 标签:程序猿一只|C++选手|学生 简介:因C语言结识编程,随后转入计算机 ...

  4. python进阶(16)深入了解GIL锁(最详细)

    前言 python的使用者都知道Cpython解释器有一个弊端,真正执行时同一时间只会有一个线程执行,这是由于设计者当初设计的一个缺陷,里面有个叫GIL锁的,但他到底是什么?我们只知道因为他导致pyt ...

  5. 开坑:mysql相关问题

    一. 先过滤后连表和先连表后在mysql中选择的哪一种? 二. left join 和inner join使用场景有什么区别? 三. 第二个问题的衍生问题:left join中where 条件使用对n ...

  6. 聊一聊Jmeter的参数化

    背景 前面一篇讲了 JMeter 的一个最简单的例子,这篇聊一下 JMeter 的参数化. 在开始之前先来一个单元测试的例子,感受一下参数化. 上面是一个用 xUnit 写的单元测试,这个单元测试就是 ...

  7. text-align: justify 文本对齐

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

  8. 【缓存池】转-Mysql参数innodb_buffer_pool_size

    转自:https://blog.csdn.net/kk185800961/article/details/79378313/ 以下考虑主要为 Innodb 引擎 [ innodb_buffer_poo ...

  9. (翻译)OpenDocument and Open XML security (OpenOffice.org and MS Office 2007)

    标题:Open Document 和 Open XML安全性(OpenOffice.org and MS Office 2007) 摘要,OpenDocument 和 Open XML 都是 Offi ...

  10. windows-CODE注入(远程线程注入)

    远程线程注入(先简单说,下面会详细说)今天整理下代码注入(远程线程注入),所谓代码注入,可以简单的理解为是在指定内进程里申请一块内存,然后把我们自己的执行代码和一些变量拷贝进去(通常是以启线程的方式) ...