• JavaSE复习_10 多线程复习


    wait()和sleep()的区别:
     1.wait():没有等待时间,而sleep()需要有等待时间作为参数。
     2.在同步中对于CPU的执行权和锁的处理不同:
      wait()会释放执行权和锁.醒来后需要重新竞争锁.
      sleep():释放执行权和,不会释放锁
    void show(){
         Synchronized(this){
              wait()          //可以同时有三个线程在此等待。只有拿到锁后,才能继续向下运行
    }
    }
    void start(){
         notifyAll();     //唤醒后,三个线程都具备了执行资格,但是并不是都具备执行权。
    }
    △interupt:可以通过interrupt强制将线程从冻结状态强制执行,但是会发生异常需要处理。
      当一个线程调用interrupt方法的时候,线程的中断状态将被置位.这是每一个线程都具备的boolean标志,每个线程都可以去检查这个标志,判断当前线程是否被中断(通过isInterrupted方法判断).如果线程被阻塞,(如wait和sleep的线程),此时调用interrupt方法,阻塞调用会被InterruptedException异常中断.如果在中断状态的时候调用sleep方法,将不会休眠,相反将清除中断状态,并且抛出InterruptedException异常.一个实例:
    public class Interrupted {
        public static void main(String[] args) {
            Task t=new Task();
            Thread t1=new Thread(t);
            t1.start();
        }
    }
    class Task implements Runnable {
    
        @Override
        public void run() {
            for (int i=0;i<50;i++) {
                if(i==30)
                    Thread.currentThread().interrupt();
                System.out.println(Thread.currentThread().isInterrupted()+""+i);
            }
        }
    }

    输出:

    false0
    false1
    false2
    false3
    false4
    false5
    false6
    false7
    false8
    false9
    false10
    false11
    false12
    false13
    false14
    false15
    false16
    false17
    false18
    false19
    false20
    false21
    false22
    false23
    false24
    false25
    false26
    false27
    false28
    false29
    true30
    true31
    true32
    true33
    true34
    true35
    true36
    true37
    true38
    true39
    true40
    true41
    true42
    true43
    true44
    true45
    true46
    true47
    true48
    true49
    △setDaemon:将线程设置为守护线程(后台线程),此时当虚拟机只剩下守护线程在运行时,守护线程将停止运行。
    △join:对于线程调用join方法,会使当前线程暂时释放执行权,值到申请加入的线程执行完毕后,该线程才会运行。
    △setPriority:设置线程的优先级
    △public static void yield():礼让线程,让出CPU的执行权.如果有其他的可运行线程具有至少与此线程同样高的优先级,那么这些线程接下来将会被调度
    △Lock.newCondition:将获取条件变量,一个锁可以有一个或多个相关的条件对象.await方法,将线程放在条件的等待集中,而signalAll方法将会解除该条件的等待集中的所有现场的阻塞状态.
    △notifyAll,wait,notify方法只能在同步方法和同步代码块中使用
    △volatile为实例域的同步实现了一种免锁机制,如果声明一个域为volatile,则编译器和虚拟机就知道该域是可能被另一个线程并发更新的.因此变量的更新操作能立即通知到其他线程,新值能立即同步到主内存.
    △可以通过ThreadLocal辅助类为每个线程构建一个自己的实例:
    可以使用下面的代码:
    public static final ThreadLocal<SimpleDateForamt> dateFormat=new ThreadLocal<SimpleDateForamt>() {
         protected SimpleDateFormat initialValue() {
              return new SimpleDateFormat("yyyy-MM-dd");
         }
    }
    当在一个给定的线程中首次调用get方法的时候会掉用initialValue方法,在此之后,get方法会返回当前线程的实例.
    判断是否会出现线程安全的问题,一定要看两点:第一,有无共享的被所有现场所操作的数据,第二,操作会不会产生安全隐患.
    △线程在调用lock()方法去试图获得锁的时候,有可能发生阻塞,tryLock方法试图申请一个锁,在成功获得锁后返回true,否则立即返回false,当前线程可以立即离开去做其他的事情.
      boolean tryLock(long time,Timeunit unit):尝试获取锁,阻塞时间不会超过给定的值,如果成功返回true.
      await方法中也可以提供参数,await(long time,Timeunit unit):如果超时的时间到了,则解除阻塞
    △Java.util.concurrent.locks定义了另外一个锁,读写锁,可以获得可以被多个读取线程所共用(但是排斥写操作)的读锁.和排斥其他所有操作的写锁.方法如下:
    private ReentrantReadWriteLock rwl=new ReentrantReadWriteLock();
    Private Lock readLock=rwl.readLock();          //读取操作共享,排斥写操作
    private Lock writeLock=rwl.writeLock();        //排斥所有其他操作
    △stop方法被停止使用,因为无法知道什么时候调用stop方法是安全的,什么时候导致对象被破坏.在希望停止线程的时候应该中断线程,被中断的线程会在安全的时候停止.(采用inInterrupted判断).
    △Callerable与Runnable类似,但是内部的方法有返回值.类型参数是返回值的类型.Future可以保存异步计算的结果,FutureTask包装器是一种很便利的机制,可以将Callable转换成Future和Runnable,它同时实现两者的接口.例如:
    import java.util.concurrent.Callable;
    import java.util.concurrent.ExecutionException;
    import java.util.concurrent.FutureTask;
    
    /**
    * Created by Administrator on 2016/6/30.
    */
    public class Exercise {
        public static void main(String[] args) {
    
            FutureTask<Integer> task=new FutureTask<Integer>(new Counter());    //包装Callable
            Thread t=new Thread(task);
            t.start();  //这是Runnable接口的特性
            try {
                System.out.println(task.get());    //这是Future接口的特性
            } catch (InterruptedException e) {
                e.printStackTrace();
            } catch (ExecutionException e) {
                e.printStackTrace();
            }
        }
    }
    class Counter implements Callable<Integer> {
        int sum;
        @Override
        public Integer call() throws Exception {
            for (int i=0;i<100;i++) {
                sum+=i;
            }
            return sum;
        }
    }
    △ExecutorService newCashedThreadPool():返回一个带缓存的线程池,该池在必要的时候创建线程.
    ExecutorService newFixedThreadPool(int threads):返回一个线程池,该池中的线程数由参数的指定,超出参数的线程将放入处理队列中.
    ExecutorService newSingleThreadExecutor():返回一个执行器,它在一个和单个的线程中依次执行各个任务.
    线程池可以调用submit提交需要执行的任务,将返回一个Future对象.可以用于取消对象.如果线程池提交的任务是Callable类型的任务可以利用返回的Future对象调用get方法获得值.
      使用连接线程池时候应该做的事情:
     1)调用Executors类中的静态方法newCachedThreadPool或newFixedThreadPool.
     2)调用submit提交Runnable和Callable对象
     3)如果想要取消一个任务或者想要获取Callable对象的值,就要保存好返回的Future对象.
     4)若不再提交任何任务的时候,调用shutdown.
    简单示例:
      
            // public static ExecutorService newFixedThreadPool(int nThreads)
            ExecutorService pool = Executors.newFixedThreadPool(2);
    
            // 可以执行Runnable对象或者Callable对象代表的线程
            Future fu1=pool.submit(new MyRunnable());
            Future fu2=pool.submit(new MyCallable());
            fu1.cancel();//取消该任务
            System.out.println(fu2.get());//获取Callable对象的运行结果            
            //结束线程池
            pool.shutdown();
  • 相关阅读:
    bzoj 3238
    bzoj 3473 后缀自动机多字符串的子串处理方法
    bzoj 2998 第k小字串
    bzoj 3672 利用点分治将CDQ分治推广到树型结构上
    bzoj 3671 贪心
    NOIP模拟题——nan
    NOIP模拟题——kun(栈)
    hduP2586——How far away ?
    DP习题
    NOIP模拟题——来自风平浪静的明天
  • 原文地址:https://www.cnblogs.com/hlhdidi/p/5631485.html
Copyright © 2020-2023  润新知