Go语言实现超时的3种方法

超时,指一个协程A开启另一个协程B,A会阻塞等待B一段指定的时间,例如:5秒,A通知B结束(也有可能不通知,让B继续运行)。也就是说,A就不愿意阻塞等待太久。

Go语言有多种方法实现这种超时,我总结出3种:

方法一:用两个通道 + A协程sleep

一个通道用来传数据,一个用来传停止信号。

package main

import (
	"fmt"
	"time"
)

// 老师视频里的生产者消费者

func main() {
	//知识点: 老师这里用了两个线程,一个用个传数据,一个用来传关闭信号
	messages := make(chan int, 10)
	done := make(chan bool)

	defer close(messages)

	// consumer
	go func() {
		ticker := time.NewTicker(1 * time.Second)
		for range ticker.C {
			select {
			case <-done:
				fmt.Println("child process interrupt...") // 数据还没收完,就被停止了。
				return
			default:
				fmt.Printf("receive message:%d\n", <-messages)
			}

		}
	}()

	// producer
	for i := 0; i < 10; i++ {
		messages <- i
	}

	// 5秒后主线程关闭done通道
	time.Sleep(5 * time.Second)
	close(done)
	time.Sleep(1 * time.Second)
	fmt.Println("main process exit!")

}

程序输出如下:

receive message:0
receive message:1
receive message:2
receive message:3
child process interrupt...
main process exit!

方法二:使用Timer(定时器)

这种方法也方法一类似,只不过是用一个Timer代替通道。

package main

import (
	"fmt"
	"time"
)

//知识点:
// 1) 多通道
// 2) 定时器
func main() {
	ch1 := make(chan int, 10)
	go func(ch chan<- int) {
		// 假设子协程j是一个耗时操作,例如访问网络,要10秒后才会有数据
		time.Sleep(10 * time.Second)
		ch <- 1
	}(ch1)

	timer := time.NewTimer(5 * time.Second) // 设置定时器的超时时间,主线程只等5秒

	fmt.Println("select start....")
	// 知识点:主协程等待子线程,并有超时机制
	select {
	case <-ch1:
		fmt.Println("从channel 1 收到一个数字")
	case <-timer.C: // 定时器也是一个通道
		fmt.Println("5秒到了,超时了,main协程不等了")
	}

	fmt.Println("done!")
}

程序输出如下:

select start....
5秒到了,超时了,main协程不等了
done!

方法三:使用context.WithTimeout

下面的例子比较复杂,基于 Channel 编写一个简单的单协程生产者消费者模型。

要求如下:

1)队列:队列长度 10,队列元素类型为 int
2)生产者:每 1 秒往队列中放入一个类型为 int 的元素,队列满时生产者可以阻塞
3)消费者:每2秒从队列中获取一个元素并打印,队列为空时消费者阻塞
4)主协程30秒后要求所有子协程退出。
5)要求优雅退出,即消费者协程退出前,要先消费完所有的int
6)通过入参支持两种运行模式:
wb(温饱模式)生产速度快过消费速度、
je(饥饿模式)生产速度慢于消费速度

context.WithTimeout见第87行。

package main

import (
	"context"
	"flag"
	"fmt"
	"sync"
	"time"
)

// 课后练习 1.2
// 基于 Channel 编写一个简单的单协程生产者消费者模型。
// 要求如下:
// 1)队列:队列长度 10,队列元素类型为 int
// 2)生产者:每 1 秒往队列中放入一个类型为 int 的元素,队列满时生产者可以阻塞
// 3)消费者:每2秒从队列中获取一个元素并打印,队列为空时消费者阻塞
// 4)主协程30秒后要求所有子协程退出。
// 5)要求优雅退出,即消费者协程退出前,要先消费完所有的int。

// 知识点:
// 1) 切片的零值也是可用的。
// 2) context.WithTimeout
var (
	wg sync.WaitGroup
	p  Producer
	c  Consumer
)

type Producer struct {
	Time     int
	Interval int
}

type Consumer struct {
	Producer
}

func (p Producer) produce(queue chan<- int, ctx context.Context) {
	go func() {
	LOOP:
		for {
			p.Time = p.Time + 1
			queue <- p.Time
			fmt.Printf("生产者进行第%d次生产,值:%d\n", p.Time, p.Time)
			time.Sleep(time.Duration(p.Interval) * time.Second)

			select {
			case <-ctx.Done():
				close(queue)
				break LOOP
			}
		}
		wg.Done()
	}()
}

func (c Consumer) consume(queue <-chan int, ctx context.Context) {
	go func() {
	LOOP:
		for {
			c.Time++
			val := <-queue
			fmt.Printf("-->消费者进行第%d次消费,值:%d\n", c.Time, val)
			time.Sleep(time.Duration(c.Interval) * time.Second)

			select {
			case <-ctx.Done():
				//remains := new([]int)
				//remains := []int{}
				var remains []int // 知识点:切片的零值也是可用的。
				for val = range queue {
					remains = append(remains, val)
					fmt.Printf("-->消费者: 最后一次消费, 值为:%v\n", remains)
					break LOOP
				}
			}
		}
		wg.Done()
	}()
}

func main() {
	wg.Add(2)

	// 知识点:context.Timeout
	timeout := 30
	ctx, _ := context.WithTimeout(context.Background(), time.Duration(timeout)*time.Second)

	queue := make(chan int, 10)

	p.produce(queue, ctx)
	fmt.Println("main waiting...")
	wg.Wait()
	fmt.Println("done")
}

/*
启动命令:
$ go run main/main.go -m wb
$ go run main/main.go -m je
*/
func init() {
	// 解析程序入参,运行模式
	mode := flag.String("m", "wb", "请输入运行模式:\nwb(温饱模式)生产速度快过消费速度、\nje(饥饿模式)生产速度慢于消费速度)")
	flag.Parse()

	p = Producer{}
	c = Consumer{}

	if *mode == "wb" {
		fmt.Println("运行模式:wb(温饱模式)生产速度快过消费速度")
		p.Interval = 1 // 每隔1秒生产一次
		c.Interval = 5 // 每隔5秒消费一次

		// p = Producer{Interval: 1}
		// c = Consumer{Interval: 5}  // 这一行会报错,为什么?

	} else {
		fmt.Println("运行模式:je(饥饿模式)生产速度慢于消费速度")
		p.Interval = 5 // 每隔5秒生产一次
		c.Interval = 1 // 每隔1秒消费一次
	}

}
wb(温饱模式)生产速度快过消费速度,输出如下:
运行模式:wb(温饱模式)生产速度快过消费速度
生产者: 第1次生产, 值为:1
-->消费者: 第1次消费, 值为:1
生产者: 第2次生产, 值为:2
生产者: 第3次生产, 值为:3
生产者: 第4次生产, 值为:4
生产者: 第5次生产, 值为:5
-->消费者: 第2次消费, 值为:2
生产者: 第6次生产, 值为:6
生产者: 第7次生产, 值为:7
生产者: 第8次生产, 值为:8
生产者: 第9次生产, 值为:9
生产者: 第10次生产, 值为:10
-->消费者: 第3次消费, 值为:3
生产者: 第11次生产, 值为:11
生产者: 第12次生产, 值为:12
生产者: 第13次生产, 值为:13
-->消费者: 第4次消费, 值为:4
生产者: 第14次生产, 值为:14
-->消费者: 第5次消费, 值为:5
生产者: 第15次生产, 值为:15
生产者: 第16次生产, 值为:16
-->消费者: 第6次消费, 值为:6
main waiting
生产者: 第17次生产, 值为:17
-->消费者: 最后一次消费, 值为:[7 8 9 10 11 12 13 14 15 16 17]
-- done --

je(饥饿模式)生产速度慢于消费速度,输出如下:

运行模式:je(饥饿模式)生产速度慢于消费速度
-->消费者: 第1次消费, 值为:1
生产者: 第1次生产, 值为:1
生产者: 第2次生产, 值为:2
-->消费者: 第2次消费, 值为:2
生产者: 第3次生产, 值为:3
-->消费者: 第3次消费, 值为:3
生产者: 第4次生产, 值为:4
-->消费者: 第4次消费, 值为:4
生产者: 第5次生产, 值为:5
-->消费者: 第5次消费, 值为:5
生产者: 第6次生产, 值为:6
-->消费者: 第6次消费, 值为:6
main waiting
-->消费者: 第7次消费, 值为:0
-->消费者: 最后一次消费, 值为:[]
-- done --

  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
### 回答1: 下面是一个使用 Go 语言实现 IP 端口连通性测试的程序: ```go package main import ( "fmt" "net" "os" "time" ) func main() { if len(os.Args) != 3 { fmt.Fprintf(os.Stderr, "Usage: %s host:port", os.Args[0]) os.Exit(1) } service := os.Args[1] timeout := time.Second conn, err := net.DialTimeout("tcp", service, timeout) if err != nil { fmt.Println("连接失败:", err) os.Exit(1) } defer conn.Close() fmt.Println("连接成功") os.Exit(0) } ``` 运行方法: ``` go run ip_port_check.go host:port ``` 在上面的代码中,`host:port` 就是要测试的 IP 地址和端口。代码使用了 `net.DialTimeout` 函数来进行连接测试,设置了超时时间为 1 秒。如果连接成功,就会输出 "连接成功";如果失败,就会输出 "连接失败" 和对应的错误信息。 ### 回答2: 使用Go语言实现IP端口连通性测试的程序可以采用以下步骤: 1. 导入相关的包:首先需要导入`net`和`time`包。 2. 获取待测试的IP地址和端口:程序可以通过命令行参数或者用户输入的方式获取待测试的IP地址和端口。 3. 创建TCP连接:使用`net.DialTimeout`函数创建TCP连接,其中可以设置一个超时时间,以防止长时间无响应。 4. 检查连接是否成功:通过检查是否有错误来确定连接是否成功。如果连接成功,则表示IP端口是可达的;如果连接出现错误,则表示IP端口无法访问。 5. 输出连接测试结果:根据连接的成功与否,输出相应的提示信息。可以使用`fmt.Println`函数打印输出连接测试结果。 下面是一个简单的示例代码实现: ```go package main import ( "fmt" "net" "os" "time" ) func main() { // 获取待测试的IP地址和端口 ip := os.Args[1] port := os.Args[2] // 创建TCP连接 address := fmt.Sprintf("%s:%s", ip, port) conn, err := net.DialTimeout("tcp", address, 3*time.Second) if err != nil { fmt.Println("连接失败:", err) return } defer conn.Close() fmt.Printf("成功连接到IP: %s 端口: %s\n", ip, port) } ``` 以上代码示例可以在命令行中使用两个参数来传递需要测试的IP地址和端口,例如: ``` go run main.go 127.0.0.1 8080 ``` 上述代码仅仅是最基本的实现,可以根据实际需求进行扩展,比如支持批量测试、并发测试等。 ### 回答3: Go语言可以使用`net`包来实现IP端口连通性测试的程序。下面是一个简单示例: ```go package main import ( "fmt" "net" "time" ) func main() { ip := "127.0.0.1" // 目标IP地址 port := 8080 // 目标端口号 address := fmt.Sprintf("%s:%d", ip, port) conn, err := net.DialTimeout("tcp", address, 5*time.Second) defer conn.Close() if err != nil { fmt.Println("连接失败:", err) return } fmt.Println("连接成功") } ``` 在上面的代码中,我们通过`DialTimeout`函数来建立TCP连接。首先,将目标IP地址和端口号格式化为地址字符串,然后通过`DialTimeout`连接到该地址。我们设置了5秒的超时时间。如果连接成功,我们输出"连接成功",否则输出连接失败的错误信息。 这个简单的示例可以帮助我们测试目标IP地址和端口的连通性。你可以根据需要进一步扩展该程序,使其更加强大和灵活。例如,你可以在一个循环中测试多个IP地址和端口号,或者可以在连接失败时进一步处理错误信息。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值