Compare commits
2 Commits
559d9730fd
...
29620426c2
Author | SHA1 | Date |
---|---|---|
|
29620426c2 | |
|
c77a5ee483 |
|
@ -0,0 +1,42 @@
|
|||
package main
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"time"
|
||||
)
|
||||
|
||||
func main() {
|
||||
// 可以把goroutine理解为轻量级的线程,
|
||||
// goroutine是由Go运行时环境管理的,
|
||||
// goroutine的调度是由Go运行时进行管理的。
|
||||
|
||||
// 当一个go程序运行时,Go运行时会为其创建一个主goroutine来执行主函数,
|
||||
// 新的goroutine使用go关键字来创建,
|
||||
// go语句是在一个普通的函数或者方法调用前加上go关键字,
|
||||
// 该函数或者方法就会在一个新的goroutine中并发执行。
|
||||
|
||||
// 并发的调用spinner函数
|
||||
// 该函数会一直打印转动的动画
|
||||
go spinner(100 * time.Millisecond)
|
||||
const n = 45
|
||||
fibN := fib(n) // slow
|
||||
fmt.Printf("\rFibonacci(%d) = %d\n", n, fibN)
|
||||
}
|
||||
|
||||
func spinner(delay time.Duration) {
|
||||
for {
|
||||
for _, r := range `-\|/` {
|
||||
// \r表示回车,不换行
|
||||
// 就是回到行首,覆盖之前的输出
|
||||
fmt.Printf("\r%c", r)
|
||||
time.Sleep(delay)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
func fib(x int) int {
|
||||
if x < 2 {
|
||||
return x
|
||||
}
|
||||
return fib(x-1) + fib(x-2)
|
||||
}
|
|
@ -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