只要了解过多线程,我们就知道线程开始的顺序跟执行的顺序是不一样的。如果只是创建三个线程然后执行,最后的执行顺序是不可预期的。这是因为在创建完线程之后,线程执行的开始时间取决于 CPU 何时分配时间片,线程可以看成是相对于的主线程的一个异步操作。

  1. public class FIFOThreadExample {
  2. public synchronized static void foo(String name) {
  3. System.out.print(name);
  4. }
  5. public static void main(String[] args) {
  6. Thread thread1 = new Thread(() -> foo("A"));
  7. Thread thread2 = new Thread(() -> foo("B"));
  8. Thread thread3 = new Thread(() -> foo("C"));
  9. thread1.start();
  10. thread2.start();
  11. thread3.start();
  12. }
  13. }

输出结果:ACB/ABC/CBA…
那么我们该如何保证线程的顺序执行呢?

如何保证线程的顺序执行?

1. 使用 Thread.join () 实现

Thread.join() 的作用是让父线程等待子线程结束之后才能继续运行。以上述例子为例,main() 方法所在的线程是父线程,在其中我们创建了 3 个子线程 A,B,C,子线程的执行相对父线程是异步的,不能保证顺序性。而对子线程使用 Thread.join() 方法之后就可以让父线程等待子线程运行结束后,再开始执行父线程,这样子线程执行被强行变成了同步的,我们用 Thread.join() 方法就能保证线程执行的顺序性。

  1. public class FIFOThreadExample {
  2. public static void foo(String name) {
  3. System.out.print(name);
  4. }
  5. public static void main(String[] args) throws InterruptedException{
  6. Thread thread1 = new Thread(() -> foo("A"));
  7. Thread thread2 = new Thread(() -> foo("B"));
  8. Thread thread3 = new Thread(() -> foo("C"));
  9. thread1.start();
  10. thread1.join();
  11. thread2.start();
  12. thread2.join();
  13. thread3.start();
  14. }
  15. }

输出结果:ABC

2. 使用单线程线程池来实现

另一种保证线程顺序执行的方法是使用一个单线程的线程池,这种线程池中只有一个线程,相应的,内部的线程会按加入的顺序来执行。

  1. import java.util.concurrent.ExecutorService;
  2. import java.util.concurrent.Executors;
  3. public class FIFOThreadExample {
  4. public static void foo(String name) {
  5. System.out.print(name);
  6. }
  7. public static void main(String[] args) throws InterruptedException{
  8. Thread thread1 = new Thread(() -> foo("A"));
  9. Thread thread2 = new Thread(() -> foo("B"));
  10. Thread thread3 = new Thread(() -> foo("C"));
  11. ExecutorService executor = Executors.newSingleThreadExecutor();
  12. executor.submit(thread1);
  13. executor.submit(thread2);
  14. executor.submit(thread3);
  15. executor.shutdown();
  16. }
  17. }

输出结果:ABC

3. 使用 volatile 关键字修饰的信号量实现

上面两种的思路都是让保证线程的执行顺序,让线程按一定的顺序执行。这里介绍第三种思路,那就是线程可以无序运行,但是执行结果按顺序执行。
你应该可以想到,三个线程都被创建并 start(),这时候三个线程随时都可能执行 run() 方法。因此为了保证 run() 执行的顺序性,我们肯定需要一个信号量来让线程知道在任意时刻能不能执行逻辑代码。
另外,因为三个线程是独立的,这个信号量的变化肯定需要对其他线程透明,因此 volatile 关键字也是必须要的。

  1. public class TicketExample2 {
  2. //信号量
  3. static volatile int ticket = 1;
  4. //线程休眠时间
  5. public final static int SLEEP_TIME = 1;
  6. public static void foo(int name){
  7. //因为线程的执行顺序是不可预期的,因此需要每个线程自旋
  8. while (true) {
  9. if (ticket == name) {
  10. try {
  11. Thread.sleep(SLEEP_TIME);
  12. //每个线程循环打印3次
  13. for (int i = 0; i < 3; i++) {
  14. System.out.println(name + " " + i);
  15. }
  16. } catch (InterruptedException e) {
  17. e.printStackTrace();
  18. }
  19. //信号量变更
  20. ticket = name%3+1;
  21. return;
  22. }
  23. }
  24. }
  25. public static void main(String[] args) throws InterruptedException {
  26. Thread thread1 = new Thread(() -> foo(1));
  27. Thread thread2 = new Thread(() -> foo(2));
  28. Thread thread3 = new Thread(() -> foo(3));
  29. thread1.start();
  30. thread2.start();
  31. thread3.start();
  32. }
  33. }

执行结果:
1 0
1 1
1 2
2 0
2 1
2 2
3 0
3 1
3 2

4. 使用 Lock 和信号量实现

此种方法的思想跟第三种方法是一样的,都是不考虑线程执行的顺序而是考虑用一些方法控制线程执行业务逻辑的顺序。这里我们同样用一个原子类型信号量 ticket,当然你可以不用原子类型,这里我只是为了保证自增操作的线程安全。然后我们用了一个可重入锁 ReentrantLock。用来给方法加锁,当一个线程拿到锁并且标识位正确的时候开始执行业务逻辑,执行完毕后唤醒下一个线程。
这里我们不需要使用 while 进行自旋操作了,因为 Lock 可以让我们唤醒指定的线程,所以改成 if 就可以实现顺序的执行。

  1. public class TicketExample3 {
  2. //信号量
  3. AtomicInteger ticket = new AtomicInteger(1);
  4. public Lock lock = new ReentrantLock();
  5. private Condition condition1 = lock.newCondition();
  6. private Condition condition2 = lock.newCondition();
  7. private Condition condition3 = lock.newCondition();
  8. private Condition[] conditions = {condition1, condition2, condition3};
  9. public void foo(int name) {
  10. try {
  11. lock.lock();
  12. //因为线程的执行顺序是不可预期的,因此需要每个线程自旋
  13. System.out.println("线程" + name + " 开始执行");
  14. if(ticket.get() != name) {
  15. try {
  16. System.out.println("当前标识位为" + ticket.get() + ",线程" + name + " 开始等待");
  17. //开始等待被唤醒
  18. conditions[name - 1].await();
  19. System.out.println("线程" + name + " 被唤醒");
  20. } catch (InterruptedException e) {
  21. e.printStackTrace();
  22. }
  23. }
  24. System.out.println(name);
  25. ticket.getAndIncrement();
  26. if (ticket.get() > 3) {
  27. ticket.set(1);
  28. }
  29. //执行完毕,唤醒下一次。1唤醒2,2唤醒3
  30. conditions[name % 3].signal();
  31. } finally {
  32. //一定要释放锁
  33. lock.unlock();
  34. }
  35. }
  36. public static void main(String[] args) throws InterruptedException {
  37. TicketExample3 example = new TicketExample3();
  38. Thread t1 = new Thread(() -> {
  39. example.foo(1);
  40. });
  41. Thread t2 = new Thread(() -> {
  42. example.foo(2);
  43. });
  44. Thread t3 = new Thread(() -> {
  45. example.foo(3);
  46. });
  47. t1.start();
  48. t2.start();
  49. t3.start();
  50. }
  51. }

输出结果:
线程 2 开始执行
当前标识位为 1, 线程 2 开始等待
线程 1 开始执行
1
线程 3 开始执行
当前标识位为 2, 线程 3 开始等待
线程 2 被唤醒
2
线程 3 被唤醒
3
上述的执行结果并非唯一,但可以保证打印的顺序一定是 123 这样的顺序。

参考文章

java 多线程 实现多个线程的顺序执行 - Hoonick - 博客园 (cnblogs.com)
Java lock 锁的一些细节_笔记小屋 - CSDN 博客
VolatileCallSite (Java Platform SE 8 ) (oracle.com)
java 保证多线程的执行顺序 - james.yj - 博客园 (cnblogs.com)