在C++中选择类似于context-select的功能。

huangapple go评论123阅读模式
英文:

context-select like features in C++

问题

想象一种情况,我想调用一个执行一些处理的函数,但是有时间限制。我可以使用context.Contextselect在golang中编写一个函数。我想象的代码如下:

  1. package main
  2. import (
  3. "context"
  4. "fmt"
  5. "time"
  6. )
  7. func longRunning(ctx context.Context, msg string) {
  8. stop := make(chan bool)
  9. done := make(chan bool)
  10. go func() {
  11. for {
  12. fmt.Printf("long running calculation %v...", msg)
  13. select {
  14. case <-stop:
  15. fmt.Println("time to stop early!")
  16. return
  17. default:
  18. }
  19. }
  20. done <- true
  21. }()
  22. select {
  23. case <-done:
  24. return
  25. case <-ctx.Done():
  26. stop <- true
  27. return
  28. }
  29. }
  30. func main() {
  31. ctx := context.Background()
  32. ctx, cancel := context.WithTimeout(ctx, 3*time.Second)
  33. defer cancel()
  34. longRunning(ctx, "wheeee")
  35. }

在C++中有没有类似的模式可以实现类似的结果?在上面的示例中,select能够以非阻塞的方式监听通道。是否可以使用某种事件文件描述符并监听事件来实现类似的功能?

任何建议或提示将不胜感激。

英文:

Imagine a situation where I'd like to call a function that does some amount of processing, but is time-bound. I could write a function in golang using context.Context and select. I'd imagine something as follows:

  1. package main
  2. import (
  3. &quot;context&quot;
  4. &quot;fmt&quot;
  5. &quot;time&quot;
  6. )
  7. func longRunning(ctx context.Context, msg string) {
  8. stop := make(chan bool)
  9. done := make(chan bool)
  10. go func() {
  11. for {
  12. fmt.Printf(&quot;long running calculation %v...&quot;, msg)
  13. select {
  14. case &lt;-stop:
  15. fmt.Println(&quot;time to stop early!&quot;)
  16. return
  17. default:
  18. }
  19. }
  20. done &lt;- true
  21. }()
  22. select {
  23. case &lt;-done:
  24. return
  25. case &lt;-ctx.Done():
  26. stop &lt;- true
  27. return
  28. }
  29. }
  30. func main() {
  31. ctx := context.Background()
  32. ctx, cancel := context.WithTimeout(ctx, 3*time.Second)
  33. defer cancel()
  34. longRunning(ctx, &quot;wheeee&quot;)
  35. }

Is there a pattern I can use to achieve similar results in C++? in the sample above select is able to listen on a channel in a non-blocking way. Is having a eventfd file descriptor of some kind and listening to events the way to do it?

Any suggestions or tips would be much appreciated.

答案1

得分: 6

大致如下所示:

  1. void longRunning(std::atomic<bool>& stop) {
  2. for (;;) {
  3. if (stop) return;
  4. // 做一些工作
  5. }
  6. }
  7. int main() {
  8. std::atomic<bool> stop = false;
  9. auto future = std::async(std::launch::async, longRunning, std::ref(stop));
  10. future.wait_for(std::chrono::seconds(num_seconds));
  11. stop = true;
  12. future.get(); // 等待任务完成或提前退出
  13. }

演示

英文:

Something along these lines, perhaps:

  1. void longRunning(std::atomic&lt;bool&gt;&amp; stop) {
  2. for (;;) {
  3. if (stop) return;
  4. // Do a bit of work
  5. }
  6. }
  7. int main() {
  8. std::atomic&lt;bool&gt; stop = false;
  9. auto future = std::async(std::launch::async, longRunning, std::ref(stop));
  10. future.wait_for(std::chrono::seconds(num_seconds));
  11. stop = true;
  12. future.get(); // wait for the task to finish or exit early.
  13. }

Demo

huangapple
  • 本文由 发表于 2021年6月20日 07:25:54
  • 转载请务必保留本文链接:https://go.coder-hub.com/68051489.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定