zoukankan      html  css  js  c++  java
  • 1114. 按序打印-多线程-简单

    问题描述

    我们提供了一个类:

    public class Foo {
      public void first() { print("first"); }
      public void second() { print("second"); }
      public void third() { print("third"); }
    }
    三个不同的线程将会共用一个 Foo 实例。

    线程 A 将会调用 first() 方法
    线程 B 将会调用 second() 方法
    线程 C 将会调用 third() 方法
    请设计修改程序,以确保 second() 方法在 first() 方法之后被执行,third() 方法在 second() 方法之后被执行。

    示例 1:

    输入: [1,2,3]
    输出: "firstsecondthird"
    解释:
    有三个线程会被异步启动。
    输入 [1,2,3] 表示线程 A 将会调用 first() 方法,线程 B 将会调用 second() 方法,线程 C 将会调用 third() 方法。
    正确的输出是 "firstsecondthird"。
    示例 2:

    输入: [1,3,2]
    输出: "firstsecondthird"
    解释:
    输入 [1,3,2] 表示线程 A 将会调用 first() 方法,线程 B 将会调用 third() 方法,线程 C 将会调用 second() 方法。
    正确的输出是 "firstsecondthird"。
     

    提示:

    尽管输入中的数字似乎暗示了顺序,但是我们并不保证线程在操作系统中的调度顺序。
    你看到的输入格式主要是为了确保测试的全面性。

    来源:力扣(LeetCode)
    链接:https://leetcode-cn.com/problems/print-in-order

    解答

    class Foo {
        public Object obj = new Object();
        public Foo() {
            
        }
        public boolean f = true, s, t;
        public void first(Runnable printFirst) throws InterruptedException {
            // printFirst.run() outputs "first". Do not change or remove this line.
            synchronized(obj){
                printFirst.run();
                obj.notifyAll();
                s = true;
            }
        }
    
        public void second(Runnable printSecond) throws InterruptedException {
            
            // printSecond.run() outputs "second". Do not change or remove this line.
            synchronized(obj){
                while(!s)obj.wait();//notify和notifyAll会使得线程在wait处返回,所以不能用if来检验,只能用while来检验
                printSecond.run();
                obj.notifyAll();
                t = true;
            }
        }
    
        public void third(Runnable printThird) throws InterruptedException {
            
            // printThird.run() outputs "third". Do not change or remove this line.
            synchronized(obj){
                while(!t)obj.wait();
                printThird.run();
            }
        }
    }

     解答2

    //这种解法,while会一直查询Done的值,直到其满足条件了。 有点浪费资源
    class Foo {
    
        public Foo() {
            
        }
        volatile int Done = 1;//不加volatile的话,线程中的while循环中的Done的值可能一直不知道内存中Done已经更新,所以一直死等。
        public void first(Runnable printFirst) throws InterruptedException {
            
            // printFirst.run() outputs "first". Do not change or remove this line.
            printFirst.run();
            Done++;
        }
    
        public void second(Runnable printSecond) throws InterruptedException {
            while(Done != 2);
            // printSecond.run() outputs "second". Do not change or remove this line.
            printSecond.run();
            Done++;
        }
    
        public void third(Runnable printThird) throws InterruptedException {
            while(Done != 3);
            // printThird.run() outputs "third". Do not change or remove this line.
            printThird.run();
        }
    }
  • 相关阅读:
    机器学习-数据与特征工程
    机器学习-聚类(clustering)算法:K-means算法
    机器学习-回归中的相关度和R平方值
    机器学习-非线性回归(Logistic Regression)及应用
    机器学习-多元线性回归(一)
    机器学习-简单线性回归(二)
    MVC,MVP,MVVM
    Git从入门到熟练
    NSNotificationCenter
    FMDB的线程安全
  • 原文地址:https://www.cnblogs.com/xxxxxiaochuan/p/13756642.html
Copyright © 2011-2022 走看看