使用Go快速开发TCP公共服务

使用Go快速开发TCP公共服务

文章目录

一、前言

之前使用的公共TCP服务无法使用了,想了一下整个实现原理不是很复杂,就利用Go快速开发了一个,利用公网服务器可以快速部署起来。

二、实现思路

在不考虑多用户使用,数据存储的情况下,总体实现思路比较简单:1、tcp服务接收tcp客户端消息;2、http服务端共享tcp服务端接收到的消息并通过接口方式提供给前端获取。提供给前端调用的方式参考了:实时通信的服务器推送机制 EventSource(SSE) (https://blog.csdn.net/DisMisPres/article/details/130539861)。

三、源码

服务端程序tcp_http_server.go:

go 复制代码
package main

import (
	"bufio"
	"fmt"
	"gopkg.in/antage/eventsource.v1"
	"log"
	"net"
	"net/http"
	"time"
)

var recvStr string

// TCP Server端测试
// 处理函数
func process(conn net.Conn) {
	defer conn.Close() // 关闭连接
	for {
		reader := bufio.NewReader(conn)
		var buf [8192]byte
		n, err := reader.Read(buf[:]) // 读取数据
		if err != nil {
			fmt.Println("read from client failed, err: ", err)
			break
		}
		recvStr = string(buf[:n])
		fmt.Println("收到Client端发来的数据:", recvStr)
	}
}

func main() {
	listen, err := net.Listen("tcp", ":8027")
	if err != nil {
		fmt.Println("Listen() failed, err: ", err)
		return
	}

	es := eventsource.New(nil, nil)
	defer es.Close()

	http.Handle("/", http.FileServer(http.Dir("./html")))
	http.Handle("/events", es)
	go func() {
		for {
			// 每2秒发送一条当前时间消息,并打印对应客户端数量
			if recvStr != "" {
				es.SendEventMessage(recvStr, "", "")
				recvStr = ""
			}
			time.Sleep(1 * time.Second)
		}
	}()

	log.Println("Open URL http://127.0.0.1:8016/ in your browser.")
	go func() {
		err := http.ListenAndServe(":8016", nil)
		if err != nil {

		}
	}()

	for {
		conn, err := listen.Accept() // 监听客户端的连接请求
		if err != nil {
			fmt.Println("Accept() failed, err: ", err)
			continue
		}
		go process(conn) // 启动一个goroutine来处理客户端的连接请求
	}
}

index.html(放在html目录下,和go中设置的文件路径对应):

html 复制代码
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>TCP服务</title>
    <script type="text/javascript">
        window.addEventListener("DOMContentLoaded", function () {
            var evsrc = new EventSource("http://127.0.0.1:8016/events");
            var msgEvent = function (ev) {
                console.log(ev.data)
                var text = ev.data;
                document.getElementById('log').innerText += text
                document.getElementById('log').innerText += "\n"
            }
            evsrc.onmessage = msgEvent;
            //evsrc.addEventListener("message", msgEvent)
            evsrc.onerror = function (ev) {
                console.log("readyState = " + ev.currentTarget.readyState);
            }
        })
    </script>
</head>
<body>
<h1>SSE test</h1>
<div>
    <ul id="log">
    </ul>
</div>
</body>
</html>

编译多平台适用的脚本build.bat:

bat 复制代码
set releasedate=%date:~0,4%%date:~5,2%%date:~8,2%0%time:~1,1%%time:~3,2%%time:~6,2%
SET CGO_ENABLED=0
SET GOARCH=amd64
SET GOOS=windows
go build -o ./bin/tcpHttpServer.exe

SET CGO_ENABLED=0
SET GOARCH=amd64
SET GOOS=linux
go build -o ./bin/tcpHttpServer

四、测试使用

我上面的端口使用情况:

tcp服务:8027端口

http服务:8016端口

index.html默认写死了调用接口:http://127.0.0.1:8016/events

服务端启动:

TCP客户端模拟发送:

服务端接收到:

浏览器查看:

五、最后

如上为demo演示,作为小工具够用了,实际项目可以做更多的扩展,比如多用户展示及使用,历史测试数据存储及展示,端口可配置,hex数据收发展示,通信方式扩展等等。

相关推荐
掘根8 小时前
【网络】高级IO——poll版本TCP服务器
网络·数据库·sql·网络协议·tcp/ip·mysql·网络安全
友友马9 小时前
『 Linux 』HTTP(一)
linux·运维·服务器·网络·c++·tcp/ip·http
_小许_11 小时前
Go语言的io输入输出流
golang
白总Server11 小时前
MySQL在大数据场景应用
大数据·开发语言·数据库·后端·mysql·golang·php
好兄弟给我起把狙13 小时前
[Golang] Select
开发语言·后端·golang
许野平13 小时前
Rust: Warp RESTful API 如何得到客户端IP?
tcp/ip·rust·restful·ip地址
不烦下雨c13 小时前
【网络】传输层协议TCP
网络·网络协议·tcp/ip
KookeeyLena514 小时前
IP池对数据爬取工作的帮助
网络·网络协议·tcp/ip
宇宙第一小趴菜20 小时前
探索网络世界:TCP/IP协议、Python Socket编程及日常生活比喻
网络·python·tcp/ip
新手嵌入式学习21 小时前
Modbus_tcp
网络·网络协议·tcp/ip