标签:使用 close pre 死循环 关闭 size code def 不能









package main
import (
"fmt"
"net" //做网络socket开发时,net包含有我们需要所有的方法和函数
_"io"
)
func process(conn net.Conn) { //类型是包.Conn
//这里我们循环的接收客户端发送的数据
defer conn.Close() //关闭conn
for {
//创建一个新的切片,每次都创建新的
buf := make([]byte, 1024) //把1024放入切片
//conn.Read(buf)
//1. 等待客户端通过conn发送信息
//2. 如果客户端没有wrtie[发送],那么协程就阻塞在这里
//fmt.Printf("服务器在等待客户端%s 发送信息\n", conn.RemoteAddr().String())
n , err := conn.Read(buf) //从conn读取
if err != nil {
fmt.Printf("客户端退出 err=%v", err)
return //break只是跳出循环,下面的代码还是可以执行的
}
//3. 显示客户端发送的内容到服务器的终端
fmt.Print(string(buf[:n])) //不写[:n]会输出后面一些乱七八糟的东西
}
}
func main() {
fmt.Println("服务器开始监听....")
//net.Listen("tcp", "0.0.0.0:8888")
//1. tcp 表示使用网络协议是tcp
//2. 0.0.0.0:8888 表示在本地监听 8888端口
listen, err := net.Listen("tcp", "0.0.0.0:8888")
if err != nil {
fmt.Println("listen err=", err)
return
}
defer listen.Close() //延时关闭listen
//循环等待客户端来链接我
for {
//等待客户端链接
fmt.Println("等待客户端来链接....")
conn, err := listen.Accept()
if err != nil { //如果监听一次,再监听第二次就会出错
fmt.Println("Accept() err=", err)
} else {
fmt.Printf("Accept() suc con=%v 客户端ip=%v\n", conn, conn.RemoteAddr().String()) //Remote远程
}
//这里准备其一个协程,为客户端服务
go process(conn) //不要在主线程做接收的工作(一个客户端就会堵住,其他人不能使用)
}
//fmt.Printf("listen suc=%v\n", listen)
}
package main
import (
"fmt"
"net"
"bufio"
"os"
"strings"
)
func main() {
conn, err := net.Dial("tcp", "172.16.1.3:8888") //Dial 拨
if err != nil {
fmt.Println("client dial err=", err)
return
}
//功能一:客户端可以发送单行数据,然后就退出
reader := bufio.NewReader(os.Stdin) //os.Stdin 代表标准输入[终端]
for {
//从终端读取一行用户输入,并准备发送给服务器
line, err := reader.ReadString(‘\n‘) // 读到一个换行就结束
if err != nil {
fmt.Println("readString err=", err)
}
//如果用户输入的是exit就退出
line = strings.Trim(line, " \r\n") //去掉 空格\r\n
if line == "exit" { //外层是死循环,需要一个退出的条件
fmt.Println("客户端退出..")
break
}
//再将line发送给服务器
_, err = conn.Write([]byte(line + "\n")) //转成byte字节
if err != nil {
fmt.Println("conn.Write err=", err)
}
}
}
标签:使用 close pre 死循环 关闭 size code def 不能
原文地址:https://www.cnblogs.com/yzg-14/p/12237402.html