我正在尝试创建一个程序,分别每隔3秒,8秒和24秒打印"Eat"
,"Work"
,"Sleep"
。这是我的代码:
package main
import (
"fmt"
"time"
)
func Remind(text string, delay time.Duration) <-chan string { //channel only for receiving strings
ch := make(chan string) // buffered/unbuffered?
go func() {
for {
msg := "The time is " + time.Now().Format("2006-01-02 15:04:05 ") + text
ch <- msg
time.Sleep(delay) // waits according to specification
}
}()
return ch
}
func main() {
ch1 := Remind("Eat", 1000*1000*1000*3) // every third second
ch2 := Remind("Work", 1000*1000*1000*8) // every eighth second
ch3 := Remind("Sleep", 1000*1000*1000*24) // every 24th second
select { // chooses one channel that is not empty. Should run forever (?)
case rem1 := <-ch1:
fmt.Println(rem1)
case rem2 := <-ch2:
fmt.Println(rem2)
case rem3 := <-ch3:
fmt.Println(rem3)
}
}
问题在于它在打印"Eat"
之后的时间后立即停止运行。在我读过的其他例子中,select
语句将永远存在。为什么现在不呢?
答案 0 :(得分:2)
我不知道你在哪里读到select
永远存在,但事实并非如此。
执行case
后,select
语句“完成”。如果case
中指定的任何通信操作都无法继续和,则没有default
分支,select
将阻止,只要任何一个com 。操作可以继续。但是,执行case
后,select
不会重复。
阅读规范中的相关部分:Select statements。
将它放在无穷无尽的for
中,让它永远重复:
for {
select { // chooses one channel that is not empty. Should run forever (?)
case rem1 := <-ch1:
fmt.Println(rem1)
case rem2 := <-ch2:
fmt.Println(rem2)
case rem3 := <-ch3:
fmt.Println(rem3)
}
}
作为旁注:
使用time.Duration
包中的常量,您可以更轻松地创建time
值:
ch1 := Remind("Eat", 3*time.Second) // every third second
ch2 := Remind("Work", 8*time.Second) // every eighth second
ch3 := Remind("Sleep", 24*time.Second) // every 24th second
您可能还想查看time.Ticker
类型,该类型适用于与Remind()
功能类似的任务。
答案 1 :(得分:2)
select
大部分类似于switch控制语句,有时也称为通信切换。 select
侦听频道上的传入数据,但也可能存在在频道上发送值的情况。在一个单词中select
用于获取或发送同时执行的goroutine的值。
在您的示例中,因为您正在主goroutine中执行当前时间,所以它总是被执行。但是因为其他goroutine在select语句中执行,所以这些并不总是有机会执行,因为一旦case
被执行,通道就会阻塞。
选择是什么:
在具有默认大小写的select语句中使用发送操作可以保证发送将是非阻塞的!
要永久运行,请在for循环中使用它:
package main
import (
"fmt"
"time"
)
func Remind(text string, delay time.Duration) <-chan string { //channel only for receiving strings
ch := make(chan string) // buffered/unbuffered?
go func() {
for {
msg := "The time is " + time.Now().Format("2006-01-02 15:04:05 ") + text
ch <- msg
time.Sleep(delay) // waits according to specification
}
}()
return ch
}
func main() {
ch1 := Remind("Eat", 1000*1000*1000*3) // every third second
ch2 := Remind("Work", 1000*1000*1000*8) // every eighth second
ch3 := Remind("Sleep", 1000*1000*1000*24) // every 24th second
for {
select { // chooses one channel that is not empty. Should run forever (?)
case rem1 := <-ch1:
fmt.Println(rem1)
case rem2 := <-ch2:
fmt.Println(rem2)
case rem3 := <-ch3:
fmt.Println(rem3)
}
}
}