Golang TCP服务器监听用户上下线

这篇博客介绍了如何使用Golang构建一个TCP服务器,详细讲述了如何监听用户上下线。在`main.go`中,服务器通过加锁的map管理客户端连接,并在接收到新连接时通知已有连接。在`user.go`文件中,每个用户对象都有一个协程用于读取数据,当读取不再阻塞且数据为空时,表示连接断开。
摘要由CSDN通过智能技术生成

main.go:

package main

import "fmt"

func main()  {
	server := NewServer("127.0.0.1",8888)
	if !server.Start() {
		fmt.Println("服务器启动失败")
		return
	}

}

server.go:

监听线程发现新连接时,把每个连接记录到map中(用锁保证数据安全),当有新连接过来时则通知已存在的连接。

package main

import (
	"fmt"
	"net"
	"sync"
)

type Server struct {
	Ip string
	Port int

	users map[string]*User
	userMutex sync.RWMutex
	msgC chan string
}

func NewServer(ip string,port int) *Server  {
	server := &Server{
		Ip: ip,
		Port: port,
		users: make(map[string]*User),
		msgC: make(chan string),
	}
	return server
}

func (this *Server) handler(conn net.Conn)  {
	fmt.Println(fmt.Sprintf("%s: 连接成功",conn.RemoteAddr().String()))
	this.userMutex.Lock()
	this.users[conn.RemoteAddr().String()] = NewUser(conn,this)
	this.userMutex.Unlock()
}

func (this *Server) Start() bool {
	listener, err := net.L
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
以下是一个简单的基于TCP协议传输文件的示例代码,使用Go语言实现: ```go package main import ( "fmt" "io" "net" "os" ) func main() { // 监听端口 listener, err := net.Listen("tcp", ":8080") if err != nil { fmt.Println("Error listening:", err.Error()) os.Exit(1) } defer listener.Close() fmt.Println("Listening on port 8080...") // 无限循环接受客户端连接 for { conn, err := listener.Accept() if err != nil { fmt.Println("Error accepting:", err.Error()) continue } // 处理客户端请求 go handleRequest(conn) } } func handleRequest(conn net.Conn) { defer conn.Close() // 读取客户端传来的文件名 fileName := make([]byte, 1024) _, err := conn.Read(fileName) if err != nil { fmt.Println("Error reading file name:", err.Error()) return } // 创建文件 file, err := os.Create(string(fileName)) if err != nil { fmt.Println("Error creating file:", err.Error()) return } defer file.Close() // 循环读取客户端传来的数据并写入文件 buf := make([]byte, 1024) for { n, err := conn.Read(buf) if err != nil { if err != io.EOF { fmt.Println("Error reading file:", err.Error()) } break } _, err = file.Write(buf[:n]) if err != nil { fmt.Println("Error writing file:", err.Error()) break } } fmt.Println("File transfer complete.") } ``` 客户端发送文件的代码可以参考以下示例: ```go package main import ( "fmt" "io" "net" "os" ) func main() { // 连接服务器 conn, err := net.Dial("tcp", "localhost:8080") if err != nil { fmt.Println("Error connecting:", err.Error()) os.Exit(1) } defer conn.Close() // 发送文件名 fileName := "test.txt" _, err = conn.Write([]byte(fileName)) if err != nil { fmt.Println("Error sending file name:", err.Error()) return } // 打开文件 file, err := os.Open(fileName) if err != nil { fmt.Println("Error opening file:", err.Error()) return } defer file.Close() // 循环读取文件并发送数据 buf := make([]byte, 1024) for { n, err := file.Read(buf) if err != nil { if err != io.EOF { fmt.Println("Error reading file:", err.Error()) } break } _, err = conn.Write(buf[:n]) if err != nil { fmt.Println("Error sending file data:", err.Error()) break } } fmt.Println("File transfer complete.") } ``` 以上代码仅供参考,实际使用时需要注意安全性和错误处理。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值