
首先,我们从一个基本的go语言tcp服务器框架开始。这个框架能够创建一个监听指定端口的tcp服务,并为每个传入的连接启动一个独立的goroutine来处理。
package main
import (
"io"
"log"
"net"
"bufio" // 引入 bufio 包
"fmt" // 引入 fmt 包
)
func main() {
// 监听TCP端口2000
srv, err := net.Listen("tcp", ":2000")
if err != nil {
log.Fatalf("无法监听端口: %v", err)
}
log.Printf("服务器正在监听端口: %s", srv.Addr().String())
defer srv.Close() // 确保在main函数退出时关闭监听器
for {
// 接受新的连接
conn, err := srv.Accept()
if err != nil {
log.Printf("接受连接失败: %v", err)
continue // 继续尝试接受下一个连接
}
// 为每个连接启动一个goroutine进行处理
go handleConnection(conn)
}
}在上述代码中,net.Listen用于创建一个TCP监听器,srv.Accept()会阻塞直到接收到一个新的客户端连接。一旦连接建立,我们就会在一个新的goroutine中调用handleConnection函数来处理该连接,从而实现并发处理多个客户端。
核心挑战在于如何从net.Conn中逐行读取数据,并将其打印到服务器的标准输出。由于net.Conn实现了io.Reader接口,我们可以利用bufio.Reader来高效地处理流式数据。bufio.Reader提供了ReadString方法,该方法可以读取直到遇到指定的分隔符(例如换行符\n)为止的字符串。
以下是handleConnection函数的具体实现:
// handleConnection 处理单个客户端连接
func handleConnection(c net.Conn) {
log.Printf("新连接来自: %s", c.RemoteAddr().String())
defer func() {
log.Printf("连接关闭: %s", c.RemoteAddr().String())
c.Close() // 确保连接在处理完成后关闭
}()
// 将 net.Conn 包装成 bufio.Reader 以便逐行读取
reader := bufio.NewReader(c)
for {
// 读取直到遇到换行符 '\n' 的字符串
line, err := reader.ReadString('\n')
if err == io.EOF {
// 客户端关闭连接
break
} else if err != nil {
// 其他读取错误
log.Printf("读取数据失败: %v", err)
break
}
// 将读取到的行打印到服务器的标准输出
fmt.Print(line)
}
}在这个handleConnection函数中:
立即学习“go语言免费学习笔记(深入)”;
将main函数和handleConnection函数组合起来,就得到了一个完整的、可运行的TCP服务器。
package main
import (
"bufio"
"fmt"
"io"
"log"
"net"
)
// handleConnection 处理单个客户端连接
func handleConnection(c net.Conn) {
log.Printf("新连接来自: %s", c.RemoteAddr().String())
defer func() {
log.Printf("连接关闭: %s", c.RemoteAddr().String())
c.Close() // 确保连接在处理完成后关闭
}()
reader := bufio.NewReader(c)
for {
line, err := reader.ReadString('\n')
if err == io.EOF {
break // 客户端关闭连接
} else if err != nil {
log.Printf("读取数据失败: %v", err)
break
}
fmt.Print(line) // 将读取到的行打印到服务器的标准输出
}
}
func main() {
srv, err := net.Listen("tcp", ":2000")
if err != nil {
log.Fatalf("无法监听端口: %v", err)
}
log.Printf("服务器正在监听端口: %s", srv.Addr().String())
defer srv.Close()
for {
conn, err := srv.Accept()
if err != nil {
log.Printf("接受连接失败: %v", err)
continue
}
go handleConnection(conn)
}
}要测试这个服务器,请按照以下步骤操作:
保存代码: 将上述代码保存为 server.go。
编译: 在终端中执行 go build -o server server.go。
运行服务器: 执行 ./server。服务器将启动并打印监听信息。
./server 2023/10/27 10:00:00 服务器正在监听端口: [::]:2000
使用Telnet连接: 打开一个新的终端,使用 telnet 命令连接到服务器。
telnet localhost 2000 Trying 127.0.0.1... Connected to localhost. Escape character is '^]'.
发送数据: 在Telnet客户端中输入文本并按回车。
test 123 foobar Another line
观察服务器输出: 在运行服务器的终端中,您将看到Telnet客户端发送的每一行数据被实时打印出来。
# (服务器终端输出) 2023/10/27 10:00:05 新连接来自: 127.0.0.1:54321 test 123 foobar Another line
虽然示例代码中包含了基本的错误处理,但在生产环境中,应进行更健壮的错误管理。例如,对于net.Listen和srv.Accept的错误,可以考虑重试机制或更详细的日志记录。在handleConnection中,除了io.EOF,其他读取错误也应有明确的策略,例如记录错误并关闭连接。
Go语言的fmt.Print系列函数在内部是带锁的,因此在多个goroutine同时调用fmt.Print时,它们会竞争锁以确保输出不会交错。这意味着即使有多个客户端同时发送数据,服务器的标准输出也不会出现乱序或部分行的情况。
然而,如果对输出的顺序有严格要求(例如,希望不同客户端的输出按连接建立时间或某种优先级顺序出现),或者希望避免频繁的锁竞争,可以考虑以下策略:
使用通道进行集中输出: 创建一个全局的字符串通道(chan string)。所有handleConnection goroutine将读取到的行发送到这个通道。然后,在一个独立的“输出goroutine”中,循环从该通道接收数据并统一打印到标准输出。这可以确保输出的顺序性,并减少fmt.Print的锁竞争。
// 示例:使用通道进行集中输出
var outputCh = make(chan string)
func init() {
// 启动一个独立的goroutine来处理所有输出
go func() {
for line := range outputCh {
fmt.Print(line)
}
}()
}
func handleConnectionWithChannel(c net.Conn) {
// ... (省略连接建立和错误处理)
reader := bufio.NewReader(c)
for {
line, err := reader.ReadString('\n')
// ... (错误处理)
outputCh <- line // 将行发送到通道
}
}defer c.Close()是一个好习惯,它确保无论handleConnection函数如何退出(正常完成或因错误),客户端连接都会被正确关闭,释放系统资源。
通过本教程,我们学习了如何利用Go语言的net包和bufio包构建一个简单的TCP服务器。该服务器能够有效地接收客户端的逐行输入,并将其实时打印到服务器的标准输出。我们探讨了bufio.Reader在处理流式数据方面的优势,并讨论了在并发场景下标准输出的同步处理策略,为构建更健壮、高效的网络应用奠定了基础。
以上就是Go语言实现TCP服务器:实时捕获客户端输入并输出到标准输出的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号