Java多线程:写一下两个线程交替打印 0~100 的奇偶数

3 篇文章 0 订阅

面试场景

面试官:Java多线程了解吗?你给我写一下,起两个线程交替打印0~100的奇偶数

小黄:啊?

面试官:就是有两个线程,一个线程打印奇数另一个打印偶数,它们交替输出。类似这样:

偶线程:0
奇线程:1
偶线程:2
……
奇线程:99
偶线程:100

小黄:啊?

面试官:……嗯。好的。回去等通知吧。

解说

遇到这种突如其来的面试题,有时候会让人无从下手。尽管可能你学习过多线程的知识,但是面试官抛一个问题过来,短时间内可能想不出如何使用这些知识来解决这个具体的问题。其实这个问题考察的知识点并不难,但是如果准备的面试的时候没有看过这道题,一时间还是比较难想出解决方案来的,而且这种题往往是让面试者手写代码

回到题目上来。首先是两个线程,其次是交替打印。这可以联系到线程之间的通信问题。这时可以想到大致的方向就是加锁,哪个线程拿到锁就打印,然后释放锁让另一个线程获取锁。两个线程轮流拿到锁,实现交替打印的效果。

起两个线程大家都会,加锁也简单,问题是如何让这两个线程轮流拿到锁呢?我们知道,加锁之后线程之间相互竞争锁,而Java默认是不保证锁的公平性的(使用公平锁可能也是一个思路),这就有可能出现同一个线程一直打印而另一个线程一直没有打印的情况。

解决方案

讨巧的方案

比较容易想的一个方案是,要输出的时候判断一下当前需要输出的数是不是自己要负责打印的值,如果是就输出,不是就直接释放锁。

 private int count = 0;
 private final Object lock = new Object();

 public void turning() {
     Thread even = new Thread(() -> {
         while (count < 100) {
             synchronized (lock) {
                 // 只处理偶数
                 if ((count & 1) == 0) {
                     System.out.println(Thread.currentThread().getName() + ": " + count++);
                 }
             }
         }
     }, "偶数");
     Thread odd = new Thread(() -> {
         while (count < 100) {
             synchronized (lock) {
                 // 只处理奇数
                 if ((count & 1) == 1) {
                     System.out.println(Thread.currentThread().getName() + ": " + count++);
                 }
             }
         }
     }, "奇数");
     even.start();
     odd.start();
}

输出结果:

偶数: 0
奇数: 1
偶数: 2
……
奇数: 99
偶数: 100

从输出上看,是实现了题目上的要求,两个线程,一个打印奇数,一个打印偶数,轮流输出。但只是用了一个讨巧的方式避开了线程交替获取锁的需求,明显没有答到面试官想考察的考点上。而且效率较低,如果同一个线程一直抢到锁,而另一个线程一直没有拿到,就会导致线程做很多无谓的空转。那么有没有更好的解决方案,让两个线程严格地交替获取到锁呢?

交替获取锁的方案

private int count = 0;
private final Object lock = new Object();

public void turning() throws InterruptedException {
   Thread even = new Thread(() -> {
       while (count <= 100) {
           synchronized (lock) {
               System.out.println("偶数: " + count++);
               lock.notifyAll();
               try {
               	   // 如果还没有结束,则让出当前的锁并休眠
                   if (count <= 100) {
                       lock.wait();
                   }
               } catch (InterruptedException e) {
                   e.printStackTrace();
               }
           }
       }
   });
   Thread odd = new Thread(() -> {
       while (count <= 100) {
           synchronized (lock) {
               System.out.println("奇数: " + count++);
               lock.notifyAll();
               try {
                   // 如果还没有结束,则让出当前的锁并休眠
                   if (count <= 100) {
                       lock.wait();
                   }
               } catch (InterruptedException e) {
                   e.printStackTrace();
               }
           }
       }
   });
   even.start();
   // 确保偶数线程线先获取到锁
   Thread.sleep(1);
   odd.start();
}

上面为了直观起见,我将两个线程都独立写了出来,其实 Thead 中的代码是相同的,可以抽成一个 Runnable 类:

public void turning() throws InterruptedException {
    new Thread(new TurningRunner(), "偶数").start();
    // 确保偶数线程线先获取到锁
    Thread.sleep(1);
    new Thread(new TurningRunner(), "奇数").start();
}

class TurningRunner implements Runnable {
  @Override
  public void run() {
      while (count <= 100) {
          // 获取锁
          synchronized (lock) {
              // 拿到锁就打印
              System.out.println(Thread.currentThread().getName() + ": " + count++);
              // 唤醒其他线程
              lock.notifyAll();
              try {
                  if (count <= 100) {
                      // 如果任务还没有结束,则让出当前的锁并休眠
                      lock.wait();
                  }
              } catch (InterruptedException e) {
                  e.printStackTrace();
              }
          }
      }
  }
}

输出结果:

偶数: 0
奇数: 1
偶数: 2
……
奇数: 99
偶数: 100

这种实现方式的原理就是线程1打印之后唤醒其他线程,然后让出锁,自己进入休眠状态。因为进入了休眠状态就不会与其他线程抢锁,此时只有线程2在获取锁,所以线程2必然会拿到锁。线程2以同样的逻辑执行,唤醒线程1并让出自己持有的锁,自己进入休眠状态。这样来来回回,持续执行直到任务完成。就达到了两个线程交替获取锁的效果了。

至此,本题解决。

扩展

两个线程交替打印的问题解决了,让我们来扩展一下,如果有三个线程,要求让它们交替输出 1、2、3,即:

线程1:1
线程2:2
线程3:3
线程1:1
线程2:2
线程3:3
……

这种情况要怎么解决呢?咱们下一篇文章再聊。

  • 15
    点赞
  • 55
    收藏
    觉得还不错? 一键收藏
  • 7
    评论
### 回答1: Java多线程交替打印可以通过使用两个线程和一个共享变量来实现。假设我们需要交替打印1到100,其中一个线程负责打印,另一个线程负责打印。 首先,我们需要创建一个共享变量count,并初始化为1。然后,我们创建两个线程,分别为奇线程和偶线程。 奇线程的任务是判断count的值是否小于等于100,如果是则打印count的值,并将count的值加1。然后,让偶线程继续执行。 偶线程的任务也是判断count的值是否小于等于100,如果是则打印count的值,并将count的值加1。然后,让奇线程继续执行。 整个过程会一直进行下去,直到count的值大于100为止。 下面是具体实现的Java代码: ```java public class AlternatePrinting { private static volatile int count = 1; private static Object lock = new Object(); public static void main(String[] args) { Thread oddThread = new Thread(new PrintOdd()); Thread evenThread = new Thread(new PrintEven()); oddThread.start(); evenThread.start(); } static class PrintOdd implements Runnable { @Override public void run() { while (count <= 100) { synchronized (lock) { if (count % 2 != 0) { System.out.println(Thread.currentThread().getName() + ": " + count); count++; lock.notifyAll(); } else { try { lock.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } } } } } static class PrintEven implements Runnable { @Override public void run() { while (count <= 100) { synchronized (lock) { if (count % 2 == 0) { System.out.println(Thread.currentThread().getName() + ": " + count); count++; lock.notifyAll(); } else { try { lock.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } } } } } } ``` 上述代码中,使用了一个共享的锁对象lock来保证奇线程和偶线程之间的互斥和同步。同时,通过wait()和notifyAll()方法实现了线程的阻塞和唤醒操作。 通过以上代码,可以实现奇和偶交替打印,输出结果类似如下: ``` Thread-0: 1 Thread-1: 2 Thread-0: 3 Thread-1: 4 ... Thread-0: 99 Thread-1: 100 ``` 这样就实现了Java多线程交替打印。 ### 回答2: Java多线程交替打印可以通过使用synchronized关键字和wait()、notify()、notifyAll()方法来实现。 首先,可以创建一个共享的对象作为锁,例如可以使用一个Object对象。 然后,创建两个线程,一个线程负责打印,另一个线程负责打印。在这两个线程的run()方法中,使用synchronized关键字来锁住共享对象。通过使用while循环来判断当前的字是否满足打印条件,如果不满足则调用wait()方法等待;如果满足则打印字并使用notify()方法唤醒其他线程。 在主线程中,使用start()方法启动这两个线程,并使用Thread.sleep()方法来控制每次打印的间隔时间。 具体实现如下: ``` public class AlternatePrinting { private static final Object lock = new Object(); private static int number = 1; public static void main(String[] args) { Thread oddThread = new Thread(() -> { while (number <= 10) { synchronized (lock) { if (number % 2 == 1) { System.out.println("奇线程:" + number++); lock.notify(); } else { try { lock.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } } } }); Thread evenThread = new Thread(() -> { while (number <= 10) { synchronized (lock) { if (number % 2 == 0) { System.out.println("偶线程:" + number++); lock.notify(); } else { try { lock.wait(); } catch (InterruptedException e) { e.printStackTrace(); } } } } }); oddThread.start(); evenThread.start(); } } ``` 上述代码会创建两个线程,一个线程负责打印,另一个线程负责打印。在主线程中启动这两个线程,程序会交替打印和偶直到字达到10为止。 ### 回答3: Java中,实现多线程交替打印可以使用两种方法:一是使用synchronized关键字,二是使用Lock和Condition接口。 方法一:使用synchronized关键字 首先,定义一个变量flag作为线程间的通信标志。然后,创建两个线程,分别使用synchronized锁住共享资源并通过wait()和notify()方法交替进行打印。具体步骤如下: 1. 创建一个实现Runnable接口的类,重run()方法。 2. 在run()方法中,使用synchronized锁住共享资源。 3. 使用while循环,判断flag的值。若flag为true,则调用wait()方法等待,否则进行打印。 4. 打印完后,将flag的值取反,并使用notify()方法唤醒其他线程。 5. 在main()方法中,创建两个线程对象,分别调用start()方法启动线程。 方法二:使用Lock和Condition接口 这种方式使用Lock接口来锁住共享资源,并使用Condition接口的await()和signal()方法来实现线程间的交替打印。具体步骤如下: 1. 创建一个实现Runnable接口的类,重run()方法。 2. 在run()方法中,使用Lock接口的lock()方法锁住共享资源。 3. 使用while循环,判断flag的值。若flag为true,则调用Condition接口的await()方法等待,否则进行打印。 4. 打印完后,将flag的值取反,并使用Condition接口的signal()方法唤醒其他线程。 5. 在main()方法中,创建一个ReentrantLock对象和两个Condition对象,分别调用lock()方法和newCondition()方法初始化。 6. 创建两个线程对象,分别调用start()方法启动线程。 这两种方法都可以实现多线程交替打印,具体选择哪种方式取决于具体的需求和场景。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值