返回介绍

solution / 1100-1199 / 1115.Print FooBar Alternately / README

发布于 2024-06-17 01:03:23 字数 3940 浏览 0 评论 0 收藏 0

1115. 交替打印 FooBar

English Version

题目描述

给你一个类:

class FooBar {
  public void foo() {
    for (int i = 0; i < n; i++) {
      print("foo");
    }
  }

  public void bar() {
    for (int i = 0; i < n; i++) {
      print("bar");
    }
  }
}

两个不同的线程将会共用一个 FooBar 实例:

  • 线程 A 将会调用 foo() 方法,而
  • 线程 B 将会调用 bar() 方法

请设计修改程序,以确保 "foobar" 被输出 n 次。

 

示例 1:

输入:n = 1
输出:"foobar"
解释:这里有两个线程被异步启动。其中一个调用 foo() 方法, 另一个调用 bar() 方法,"foobar" 将被输出一次。

示例 2:

输入:n = 2
输出:"foobarfoobar"
解释:"foobar" 将被输出两次。

 

提示:

  • 1 <= n <= 1000

解法

方法一:多线程 + 信号量

我们用两个信号量 $f$ 和 $b$ 来控制两个线程的执行顺序,其中 $f$ 初始值为 $1$,而 $b$ 初始值为 $0$,表示线程 $A$ 先执行。

当线程 $A$ 执行时,首先会执行 $f$ 的 $acquire$ 操作,此时 $f$ 的值变为 $0$,线程 $A$ 获得了 $f$ 的使用权,可以执行 $foo$ 函数,然后执行 $b$ 的 $release$ 操作,此时 $b$ 的值变为 $1$,线程 $B$ 获得了 $b$ 的使用权,可以执行 $bar$ 函数。

当线程 $B$ 执行时,首先会执行 $b$ 的 $acquire$ 操作,此时 $b$ 的值变为 $0$,线程 $B$ 获得了 $b$ 的使用权,可以执行 $bar$ 函数,然后执行 $f$ 的 $release$ 操作,此时 $f$ 的值变为 $1$,线程 $A$ 获得了 $f$ 的使用权,可以执行 $foo$ 函数。

因此,我们只需要循环 $n$ 次,每次执行 $foo$ 和 $bar$ 函数时,先执行 $acquire$ 操作,再执行 $release$ 操作即可。

时间复杂度 $O(n)$,空间复杂度 $O(1)$。

from threading import Semaphore


class FooBar:
  def __init__(self, n):
    self.n = n
    self.f = Semaphore(1)
    self.b = Semaphore(0)

  def foo(self, printFoo: "Callable[[], None]") -> None:
    for _ in range(self.n):
      self.f.acquire()
      # printFoo() outputs "foo". Do not change or remove this line.
      printFoo()
      self.b.release()

  def bar(self, printBar: "Callable[[], None]") -> None:
    for _ in range(self.n):
      self.b.acquire()
      # printBar() outputs "bar". Do not change or remove this line.
      printBar()
      self.f.release()
class FooBar {
  private int n;
  private Semaphore f = new Semaphore(1);
  private Semaphore b = new Semaphore(0);

  public FooBar(int n) {
    this.n = n;
  }

  public void foo(Runnable printFoo) throws InterruptedException {
    for (int i = 0; i < n; i++) {
      f.acquire(1);
      // printFoo.run() outputs "foo". Do not change or remove this line.
      printFoo.run();
      b.release(1);
    }
  }

  public void bar(Runnable printBar) throws InterruptedException {
    for (int i = 0; i < n; i++) {
      b.acquire(1);
      // printBar.run() outputs "bar". Do not change or remove this line.
      printBar.run();
      f.release(1);
    }
  }
}
#include <semaphore.h>

class FooBar {
private:
  int n;
  sem_t f, b;

public:
  FooBar(int n) {
    this->n = n;
    sem_init(&f, 0, 1);
    sem_init(&b, 0, 0);
  }

  void foo(function<void()> printFoo) {
    for (int i = 0; i < n; i++) {
      sem_wait(&f);
      // printFoo() outputs "foo". Do not change or remove this line.
      printFoo();
      sem_post(&b);
    }
  }

  void bar(function<void()> printBar) {
    for (int i = 0; i < n; i++) {
      sem_wait(&b);
      // printBar() outputs "bar". Do not change or remove this line.
      printBar();
      sem_post(&f);
    }
  }
};

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文