day10协程 练习8.1
parent
c77a5ee483
commit
29620426c2
|
@ -0,0 +1,94 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// 定义一个*data类型切片
|
||||||
|
var dSlice []*data
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
// 1. 从控制台读取要记录的时区服务器地址
|
||||||
|
// 2. 启动一个goroutine,每隔一秒钟向服务器发送一个请求,获取当前时间
|
||||||
|
// 3. 将服务器返回的时间输出到控制台
|
||||||
|
|
||||||
|
var wg sync.WaitGroup
|
||||||
|
wg.Add(len(os.Args[1:]))
|
||||||
|
// 获取服务器地址
|
||||||
|
// Tokyo=localhost:8020 London=localhost:8030
|
||||||
|
var server = make(map[int][]string)
|
||||||
|
for i, v := range os.Args[1:] {
|
||||||
|
// 处理输入
|
||||||
|
server[i] = strings.Split(v, "=")
|
||||||
|
}
|
||||||
|
// 启动时钟
|
||||||
|
for i, server := range server {
|
||||||
|
// 创建一个data类型的指针
|
||||||
|
dSlice = append(dSlice, new(data))
|
||||||
|
// 启动时钟
|
||||||
|
go clock(server[1], dSlice[i], &wg)
|
||||||
|
}
|
||||||
|
// 输出
|
||||||
|
// $ go run 8.1.go China=localhost:8000 Tokyo=localhost:8010
|
||||||
|
//====================================
|
||||||
|
//Tokyo localhost:8010 13:55:32
|
||||||
|
//China localhost:8000 12:55:32
|
||||||
|
|
||||||
|
// 输出前等待1s, 保证数据已经接收到并且写入到data类型的指针中
|
||||||
|
time.Sleep(1 * time.Second)
|
||||||
|
for {
|
||||||
|
fmt.Println("====================================")
|
||||||
|
for i, s := range server {
|
||||||
|
go output(s, dSlice[i])
|
||||||
|
}
|
||||||
|
time.Sleep(1 * time.Second)
|
||||||
|
}
|
||||||
|
// 尽管上面的死循环会阻塞主协程
|
||||||
|
// 但是还是写上吧
|
||||||
|
// 调试也方便
|
||||||
|
wg.Wait()
|
||||||
|
}
|
||||||
|
func output(ser []string, data *data) {
|
||||||
|
fmt.Println(ser[0], ser[1], data)
|
||||||
|
}
|
||||||
|
|
||||||
|
// data 类型
|
||||||
|
type data string
|
||||||
|
|
||||||
|
// 输出*data的时候,直接输出解引用data的值
|
||||||
|
func (d *data) String() string {
|
||||||
|
return string(*d)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 实现Writer接口, 用于将数据写入到data类型的指针中
|
||||||
|
func (d *data) Write(p []byte) (n int, err error) {
|
||||||
|
// 写入数据
|
||||||
|
*d = data(p)
|
||||||
|
//fmt.Println(d)
|
||||||
|
return len(p), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// 编写监听多个服务器的时钟函数
|
||||||
|
func clock(server string, data *data, wg *sync.WaitGroup) {
|
||||||
|
defer wg.Done()
|
||||||
|
//fmt.Println(server)
|
||||||
|
conn, err := net.Dial("tcp", server)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
defer conn.Close()
|
||||||
|
mustCopy(data, conn)
|
||||||
|
|
||||||
|
}
|
||||||
|
func mustCopy(dst io.Writer, src io.Reader) {
|
||||||
|
if _, err := io.Copy(dst, src); err != nil {
|
||||||
|
log.Fatalln(err)
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,57 @@
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
// go run exampleClock.go -p 8010 -loc Asia/Tokyo
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
// 设置flag
|
||||||
|
locstr := flag.String("loc", "Asia/Shanghai", "请输入完整的时区名称")
|
||||||
|
listenstr := flag.String("p", "8000", "请输入监听端口")
|
||||||
|
flag.Parse()
|
||||||
|
// 设置时区
|
||||||
|
loc, err := time.LoadLocation(*locstr)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 监听8000端口
|
||||||
|
listener, err := net.Listen("tcp", ":"+*listenstr)
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
for {
|
||||||
|
conn, err := listener.Accept()
|
||||||
|
if err != nil {
|
||||||
|
log.Print(err) // e.g., connection aborted
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
//handleConn(conn) // handle one connection at a time
|
||||||
|
// 在调用handleConn时加上go关键字,就可以同时处理多个客户端连接
|
||||||
|
go handleConn(conn, loc) // handle connections concurrently
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
var num = 0
|
||||||
|
|
||||||
|
func handleConn(c net.Conn, loc *time.Location) {
|
||||||
|
num++
|
||||||
|
defer c.Close()
|
||||||
|
fmt.Println("handleConn", num)
|
||||||
|
for {
|
||||||
|
_, err := io.WriteString(c, time.Now().In(loc).Format("15:04:05"))
|
||||||
|
if err != nil {
|
||||||
|
return // e.g., client disconnected
|
||||||
|
}
|
||||||
|
time.Sleep(1 * time.Second)
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,26 @@
|
||||||
|
// Netcat1 is a read-only TCP client.
|
||||||
|
package main
|
||||||
|
|
||||||
|
import (
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
"net"
|
||||||
|
"os"
|
||||||
|
)
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
// 连接到8000端口
|
||||||
|
conn, err := net.Dial("tcp", "localhost:8000")
|
||||||
|
if err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
defer conn.Close()
|
||||||
|
// 从标准输入读取数据
|
||||||
|
mustCopy(os.Stdout, conn)
|
||||||
|
}
|
||||||
|
|
||||||
|
func mustCopy(dst io.Writer, src io.Reader) {
|
||||||
|
if _, err := io.Copy(dst, src); err != nil {
|
||||||
|
log.Fatal(err)
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue