Go语言基于Socket编写服务器端与客户端通信的实例


Posted in Python onFebruary 19, 2016

在golang中,网络协议已经被封装的非常完好了,想要写一个Socket的Server,我们并不用像其他语言那样需要为socket、bind、listen、receive等一系列操作头疼,只要使用Golang中自带的net包即可很方便的完成连接等操作~
在这里,给出一个最最基础的基于Socket的Server的写法:

package main  

import (  

    "fmt"  

    "net"  

    "log"  

    "os"  

)  

  

  

func main() {  

  

//建立socket,监听端口  

    netListen, err := net.Listen("tcp", "localhost:1024")  

    CheckError(err)  

    defer netListen.Close()  

  

    Log("Waiting for clients")  

    for {  

        conn, err := netListen.Accept()  

        if err != nil {  

            continue  

        }  

  

        Log(conn.RemoteAddr().String(), " tcp connect success")  

        handleConnection(conn)  

    }  

}  

//处理连接  

func handleConnection(conn net.Conn) {  

  

    buffer := make([]byte, 2048)  

  

    for {  

  

        n, err := conn.Read(buffer)  

  

        if err != nil {  

            Log(conn.RemoteAddr().String(), " connection error: ", err)  

            return  

        }  

  

  

        Log(conn.RemoteAddr().String(), "receive data string:\n", string(buffer[:n]))  

  

    }  

  

}  

func Log(v ...interface{}) {  

    log.Println(v...)  

}  

  

func CheckError(err error) {  

    if err != nil {  

        fmt.Fprintf(os.Stderr, "Fatal error: %s", err.Error())  

        os.Exit(1)  

    }  

} 

唔,抛除Go语言里面10行代码有5行error的蛋疼之处,你可以看到,Server想要建立并接受一个Socket,其核心流程就是

netListen, err := net.Listen("tcp", "localhost:1024") 
conn, err := netListen.Accept() 
n, err := conn.Read(buffer) 

这三步,通过Listen、Accept 和Read,我们就成功的绑定了一个端口,并能够读取从该端口传来的内容~
Server写好之后,接下来就是Client方面啦,我手写一个HelloWorld给大家:
package main  

  

import (  

    "fmt"  

    "net"  

    "os"  

)  

  

func sender(conn net.Conn) {  

        words := "hello world!"  

        conn.Write([]byte(words))  

    fmt.Println("send over")  

  

}  

  

  

  

func main() {  

    server := "127.0.0.1:1024"  

    tcpAddr, err := net.ResolveTCPAddr("tcp4", server)  

    if err != nil {  

        fmt.Fprintf(os.Stderr, "Fatal error: %s", err.Error())  

        os.Exit(1)  

    }  

  

    conn, err := net.DialTCP("tcp", nil, tcpAddr)  

    if err != nil {  

        fmt.Fprintf(os.Stderr, "Fatal error: %s", err.Error())  

        os.Exit(1)  

    }  

  

  

    fmt.Println("connect success")  

    sender(conn)  

  

} 

可以看到,Client这里的关键在于
tcpAddr, err := net.ResolveTCPAddr("tcp4", server) 
conn, err := net.DialTCP("tcp", nil, tcpAddr) 

这两步,主要是负责解析端口和连接~
写好Server和Client之后,让我们运行一下看看:~~
成功运行,Console出现Server等待连接的提示:

Go语言基于Socket编写服务器端与客户端通信的实例

Go语言基于Socket编写服务器端与客户端通信的实例

Go语言基于Socket编写服务器端与客户端通信的实例

Server端成功的收到了我们的Hello-World啦,至于后面的那行红字,则是断开连接的提示~

到这里,一个最基础的使用Socket的Server-Client框架就出来啦~
如果想要让Server能够响应来自不同Client的请求,我们只要在Server端的代码的main入口中,
在 handleConnection(conn net.Conn) 这句代码的前面加上一个 go,就可以让服务器并发处理不同的Client发来的请求啦

自定义通讯协议
在上面我们做出来一个最基础的demo后,已经可以初步实现Server和Client之间的信息交流了~ 这一章我会介绍一下怎么在Server和Client之间实现一个简单的通讯协议,从而增强整个信息交流过程的稳定性。
        在Server和client的交互过程中,有时候很难避免出现网络波动,而在通讯质量较差的时候,Client有可能无法将信息流一次性完整发送,最终传到Server上的信息很可能变为很多段。
        如下图所示,本来应该是分条传输的json,结果因为一些原因连接在了一起,这时候就会出现问题啦,Server端要怎么判断收到的消息是否完整呢?~

Go语言基于Socket编写服务器端与客户端通信的实例

 唔,答案就是这篇文章的主题啦:在Server和Client交互的时候,加入一个通讯协议(protocol),让二者的交互通过这个协议进行封装,从而使Server能够判断收到的信息是否为完整的一段。(也就是解决分包的问题)
        因为主要目的是为了让Server能判断客户端发来的信息是否完整,因此整个协议的核心思路并不是很复杂:
协议的核心就是设计一个头部(headers),在Client每次发送信息的时候将header封装进去,再让Server在每次收到信息的时候按照预定格式将消息进行解析,这样根据Client传来的数据中是否包含headers,就可以很轻松的判断收到的信息是否完整了~
        如果信息完整,那么就将该信息发送给下一个逻辑进行处理,如果信息不完整(缺少headers),那么Server就会把这条信息与前一条信息合并继续处理。

        下面是协议部分的代码,主要分为数据的封装(Enpack)和解析(Depack)两个部分,其中Enpack用于Client端将传给服务器的数据封装,而Depack是Server用来解析数据,其中Const部分用于定义Headers,HeaderLength则是Headers的长度,用于后面Server端的解析。这里要说一下ConstMLength,这里代表Client传入信息的长度,因为在golang中,int转为byte后会占4长度的空间,因此设定为4。每次Client向Server发送信息的时候,除了将Headers封装进去意以外,还会将传入信息的长度也封装进去,这样可以方便Server进行解析和校验。

//通讯协议处理  

package protocol  

  

import (  

    "bytes"  

    "encoding/binary"  

)  

const (  

    ConstHeader         = "Headers"  

    ConstHeaderLength   = 7  

    ConstMLength = 4  

)  

  

//封包  

func Enpack(message []byte) []byte {  

    return append(append([]byte(ConstHeader), IntToBytes(len(message))...), message...)  

}  

  

//解包  

func Depack(buffer []byte, readerChannel chan []byte) []byte {  

    length := len(buffer)  

  

    var i int  

    for i = 0; i < length; i = i + 1 {  

        if length < i+ConstHeaderLength+ConstMLength {  

            break  

        }  

        if string(buffer[i:i+ConstHeaderLength]) == ConstHeader {  

            messageLength := BytesToInt(buffer[i+ConstHeaderLength : i+ConstHeaderLength+ConstMLength])  

            if length < i+ConstHeaderLength+ConstLength+messageLength {  

                break  

            }  

            data := buffer[i+ConstHeaderLength+ConstMLength : i+ConstHeaderLength+ConstMLength+messageLength]  

            readerChannel <- data  

  

        }  

    }  

  

    if i == length {  

        return make([]byte, 0)  

    }  

    return buffer[i:]  

}  

  

//整形转换成字节  

func IntToBytes(n int) []byte {  

    x := int32(n)  

  

    bytesBuffer := bytes.NewBuffer([]byte{})  

    binary.Write(bytesBuffer, binary.BigEndian, x)  

    return bytesBuffer.Bytes()  

}  

  

//字节转换成整形  

func BytesToInt(b []byte) int {  

    bytesBuffer := bytes.NewBuffer(b)  

  

    var x int32  

    binary.Read(bytesBuffer, binary.BigEndian, &x)  

  

    return int(x)  

} 

        协议写好之后,接下来就是在Server和Client的代码中应用协议啦,下面是Server端的代码,主要负责解析Client通过协议发来的信息流:
package main    

    

import (    

    "protocol"    

    "fmt"    

    "net"    

    "os"    

)    

    

func main() {    

    netListen, err := net.Listen("tcp", "localhost:6060")    

    CheckError(err)    

    

    defer netListen.Close()    

    

    Log("Waiting for clients")    

    for {    

        conn, err := netListen.Accept()    

        if err != nil {    

            continue    

        }    

    

        //timeouSec :=10    

        //conn.    

        Log(conn.RemoteAddr().String(), " tcp connect success")    

        go handleConnection(conn)    

    

    }    

}    

    

func handleConnection(conn net.Conn) {    

    

    

    // 缓冲区,存储被截断的数据    

    tmpBuffer := make([]byte, 0)    

    

    //接收解包    

    readerChannel := make(chan []byte, 16)    

    go reader(readerChannel)    

    

    buffer := make([]byte, 1024)    

    for {    

    n, err := conn.Read(buffer)    

    if err != nil {    

    Log(conn.RemoteAddr().String(), " connection error: ", err)    

    return    

    }    

    

    tmpBuffer = protocol.Depack(append(tmpBuffer, buffer[:n]...), readerChannel)    

    }    

    defer conn.Close()    

}    

    

func reader(readerChannel chan []byte) {    

    for {    

        select {    

        case data := <-readerChannel:    

            Log(string(data))    

        }    

    }    

}    

    

func Log(v ...interface{}) {    

    fmt.Println(v...)    

}    

    

func CheckError(err error) {    

    if err != nil {    

        fmt.Fprintf(os.Stderr, "Fatal error: %s", err.Error())    

        os.Exit(1)    

    }    

}   

        然后是Client端的代码,这个简单多了,只要给信息封装一下就可以了~:

package main    

import (    

"protocol"    

"fmt"    

"net"    

"os"    

"time"    

"strconv"    

    

)    

    

func send(conn net.Conn) {    

    for i := 0; i < 100; i++ {    

        session:=GetSession()    

        words := "{\"ID\":"+ strconv.Itoa(i) +"\",\"Session\":"+session +"2015073109532345\",\"Meta\":\"golang\",\"Content\":\"message\"}"    

        conn.Write(protocol.Enpacket([]byte(words)))    

    }    

    fmt.Println("send over")    

    defer conn.Close()    

}    

    

func GetSession() string{    

    gs1:=time.Now().Unix()    

    gs2:=strconv.FormatInt(gs1,10)    

    return gs2    

}    

    

func main() {    

    server := "localhost:6060"    

    tcpAddr, err := net.ResolveTCPAddr("tcp4", server)    

    if err != nil {    

        fmt.Fprintf(os.Stderr, "Fatal error: %s", err.Error())    

        os.Exit(1)    

    }    

    

    conn, err := net.DialTCP("tcp", nil, tcpAddr)    

    if err != nil {    

        fmt.Fprintf(os.Stderr, "Fatal error: %s", err.Error())    

        os.Exit(1)    

    }    

    

    

    fmt.Println("connect success")    

    send(conn)    

    

    

    

}   

这样我们就成功实现在Server和Client之间建立一套自定义的基础通讯协议啦,让我们运行一下看下效果:

Go语言基于Socket编写服务器端与客户端通信的实例

成功识别每一条Client发来的信息啦~~

Python 相关文章推荐
python中函数传参详解
Jul 03 Python
python如何查看系统网络流量的信息
Sep 12 Python
python中numpy的矩阵、多维数组的用法
Feb 05 Python
python中多个装饰器的执行顺序详解
Oct 08 Python
python实现名片管理系统
Nov 29 Python
Python面向对象程序设计类变量与成员变量、类方法与成员方法用法分析
Apr 12 Python
Python中判断子串存在的性能比较及分析总结
Jun 23 Python
numpy.transpose()实现数组的转置例子
Dec 02 Python
Python使用QQ邮箱发送邮件实例与QQ邮箱设置详解
Feb 18 Python
Python 使用 PyQt5 开发的关机小工具分享
Jul 16 Python
Python 线程池模块之多线程操作代码
May 20 Python
基于Python编写一个监控CPU的应用系统
Jun 25 Python
使用C#配合ArcGIS Engine进行地理信息系统开发
Feb 19 #Python
Python中使用OpenCV库来进行简单的气象学遥感影像计算
Feb 19 #Python
Python实现以时间换空间的缓存替换算法
Feb 19 #Python
Python使用爬虫猜密码
Feb 19 #Python
使用Python简单的实现树莓派的WEB控制
Feb 18 #Python
在Ubuntu系统下安装使用Python的GUI工具wxPython
Feb 18 #Python
以一个投票程序的实例来讲解Python的Django框架使用
Feb 18 #Python
You might like
php 数组二分法查找函数代码
2010/02/16 PHP
Zend Studio 实用快捷键一览表(精心整理)
2013/08/10 PHP
php调用nginx的mod_zip模块打包ZIP文件
2014/06/11 PHP
ThinkPHP3.1.3版本新特性概述
2014/06/19 PHP
php从数据库查询结果生成树形列表的方法
2015/04/17 PHP
一个简单安全的PHP验证码类 附调用方法
2016/06/24 PHP
php使用gd2绘制基本图形示例(直线、圆、正方形)
2017/02/15 PHP
Laravel框架下的Contracts契约详解
2020/03/17 PHP
javascript模拟的Ping效果代码 (Web Ping)
2011/03/13 Javascript
jquery+json实现动态商品内容展示的方法
2016/01/14 Javascript
微信端开发--登录小程序步骤
2017/01/11 Javascript
vue项目中v-model父子组件通信的实现详解
2017/12/10 Javascript
vue中如何让子组件修改父组件数据
2018/06/14 Javascript
axios携带cookie配置详解(axios+koa)
2018/12/28 Javascript
Vue 路由间跳转和新开窗口的方式(query、params)
2019/12/25 Javascript
js加减乘除精确运算方法实例代码
2021/01/17 Javascript
[04:45]DOTA2上海特级锦标赛主赛事第四日RECAP
2016/03/06 DOTA
使用FastCGI部署Python的Django应用的教程
2015/07/22 Python
基于python的Tkinter实现一个简易计算器
2015/12/31 Python
Python解析json文件相关知识学习
2016/03/01 Python
Tensorflow中的placeholder和feed_dict的使用
2018/07/09 Python
python opencv实现运动检测
2018/07/10 Python
python对csv文件追加写入列的方法
2019/08/01 Python
python图形开发GUI库pyqt5的详细使用方法及各控件的属性与方法
2020/02/14 Python
Python解析微信dat文件的方法
2020/11/30 Python
Python中过滤字符串列表的方法
2020/12/22 Python
CSS3 绘制BMW logo实的现代码
2013/04/25 HTML / CSS
实例教程 一款纯css3实现的数字统计游戏
2014/11/10 HTML / CSS
美国最大的珠宝首饰网上商城:Jewelry.com
2016/07/22 全球购物
美国著名珠宝品牌之一:Jared The Galleria Of Jewelry
2016/10/01 全球购物
AJAX检测用户名是否存在的方法
2021/03/24 Javascript
医疗纠纷协议书
2014/04/16 职场文书
餐厅收银员岗位职责
2015/04/07 职场文书
2015年检验科工作总结
2015/04/27 职场文书
放牛班的春天观后感
2015/06/01 职场文书
Python 用户输入和while循环的操作
2021/05/23 Python