分享
  1. 首页
  2. 文章

一个golang 实现tcp服务例子

何吕 · · 7528 次点击 · · 开始浏览
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

package main
import (
 "fmt"
 "net"
 "time"
 "os"
 "os/signal"
 "sync"
)
const (
 WorkerCount = 2
)
type Task struct {
 Id int32
 Message string
}
var wg sync.WaitGroup
var taskChannel = make(chan Task)
var signChannel = make(chan os.Signal, 1)
var exitChanel = make(chan int)
func main() {
 go installSign()
 tcpAddr, err := net.ResolveTCPAddr("tcp4", "127.0.0.1:2202")
 if err != nil {
 panic("解析ip地址失败: " + err.Error())
 }
 fmt.Println("Listening 127.0.0.1:2202 ....")
 listener, err := net.ListenTCP("tcp", tcpAddr)
 if err != nil {
 panic("监听TCP失败: " + err.Error())
 }
 fmt.Println("Listen success on 127.0.0.1:2202 with tcp4")
 defer func() {
 fmt.Println("Close listenning ....")
 listener.Close()
 fmt.Println("Shutdown")
 }()
 connChannel := make(chan net.Conn)
 go accept(listener, connChannel)
 go handleConn(connChannel)
 go taskDispatch()
 for {
 select {
 case <- signChannel:
 fmt.Println("Get shutdown sign")
 go notifyGoroutingExit()
 goto EXIT
 }
 }
 EXIT:
 fmt.Println("Waiting gorouting exit ....")
 wg.Wait()
}
func accept(listener * net.TCPListener, connChannel chan net.Conn) {
 for {
 connection, err := listener.AcceptTCP()
 if err != nil {
 fmt.Println("Accept 失败: " + err.Error())
 } else {
 connChannel <- connection
 }
 }
}
func handleConn(connChannel chan net.Conn) {
 fmt.Println("Wating connection ....")
 for {
 select {
 case conn := <- connChannel:
 remoteAddr := conn.RemoteAddr()
 fmt.Println("Client " + remoteAddr.String() + " connected")
 readConn(&conn)
 }
 }
}
func readConn(conn *net.Conn) {
 for {
 (*conn).SetReadDeadline(time.Now().Add(5 * time.Second))
 buf := make([]byte, 1024)
 _, err := (*conn).Read(buf)
 if err != nil {
 fmt.Println("Read connection error: " + err.Error())
 if err.Error() == "EOF" {
 (*conn).Close();
 fmt.Println("Close connection " + (*conn).RemoteAddr().String())
 break
 }
 }
 if buf != nil {
 fmt.Printf("Read message from connect: %s", string(buf))
 writeConn(conn, buf)
 var task Task
 task.Id = 1
 task.Message = string(buf)
 taskChannel <- task
 }
 }
}
func writeConn(conn *net.Conn, msg []byte) {
 _, err := (*conn).Write(msg)
 if err != nil {
 fmt.Println("Write connection error: " + err.Error())
 if err.Error() == "EOF" {
 (*conn).Close();
 fmt.Println("Close connection " + (*conn).RemoteAddr().String())
 }
 }
}
func taskDispatch() {
 fmt.Println("Init task moniter ....")
 for i := 0; i < WorkerCount; i ++ {
 go loop()
 }
 fmt.Println("Init task moniter DONE!")
}
func loop() {
 ticker := time.NewTicker(10 * time.Second)
 wg.Add(1)
 defer func() {
 defer wg.Done()
 defer ticker.Stop()
 }()
 for {
 fmt.Println("Wating task ....")
 select {
 case task := <- taskChannel:
 fmt.Println("Task comming: " + task.Message)
 break;
 case <- exitChanel:
 fmt.Println("Woker get exit sign")
 goto STOP
 //default:
 }
 // Epoll, 去读任务数据, 不需要处理超时的情况
 //select {
 // case <- ticker.C:
 // fmt.Println(time.Now().String() + " No task after 10 second")
 // break;
 //}
 }
 STOP:
 //TODO: Clear undo task
}
func installSign() {
 signal.Notify(signChannel, os.Interrupt, os.Kill)
}
func notifyGoroutingExit() {
 for i := 0; i < WorkerCount; i ++ {
 exitChanel <- 1
 }
}

有疑问加站长微信联系(非本文作者)

本文来自:开源中国博客

感谢作者:何吕

查看原文:一个golang 实现tcp服务例子

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

关注微信
7528 次点击
1 回复 | 直到 2018年02月01日 14:48:20
暂无回复
添加一条新回复 (您需要 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传

用户登录

没有账号?注册
(追記) (追記ここまで)

今日阅读排行

    加载中
(追記) (追記ここまで)

一周阅读排行

    加载中

关注我

  • 扫码关注领全套学习资料 关注微信公众号
  • 加入 QQ 群:
    • 192706294(已满)
    • 731990104(已满)
    • 798786647(已满)
    • 729884609(已满)
    • 977810755(已满)
    • 815126783(已满)
    • 812540095(已满)
    • 1006366459(已满)
    • 692541889

  • 关注微信公众号
  • 加入微信群:liuxiaoyan-s,备注入群
  • 也欢迎加入知识星球 Go粉丝们(免费)

给该专栏投稿 写篇新文章

每篇文章有总共有 5 次投稿机会

收入到我管理的专栏 新建专栏