线程调度是指按照特定机制为多个线程分配CPU的使用权。

有两种调度模型:分时调度模型和抢占式调度模型。  

分时调度模型:是指让所有的线程轮流获得cpu的使用权,并且平均分配每个线程占用的CPU的时间片。

抢占式调度模型:是指优先让可运行池中优先级高的线程占用CPU,如果可运行池中的线程优先级相同,那么就随机选择一个线程,使其占用CPU。处于运行状态的线程会一直运行,直至它不得不放弃CPU。一个线程会因为以下原因而放弃CPU:

1 、java虚拟机让当前线程暂时放弃CPU,转到就绪状态,使其它线程或者运行机会。

2、 当前线程因为某些原因而进入阻塞状态

3、 线程结束运行

需要注意的是,线程的调度不是跨平台的,它不仅仅取决于java虚拟机,还依赖于操作系统。在某些操作系统中,只要运行中的线程没有遇到阻塞,就不会放弃CPU;在某些操作系统中,即使线程没有遇到阻塞,也会运行一段时间后放弃CPU,给其它线程运行的机会,java的线程调度是不分时的,同时启动多个线程后,不能保证各个线程轮流获得均等的CPU时间片。如果希望明确地让一个线程给另外一个线程运行的机会,可以采取以下办法之一:调整各个线程的优先级、让处于运行状态的线程调用、Thread.sleep()方法、让处于运行状态的线程调用Thread.yield()方法、让处于运行状态的线程调用另一个线程的join()方法

所以,让多个线程顺序执行有三个确切的办法:

1、改变线程的优先级(不同的系统上可能并不会按照优先级来执行)

2、调用join()方法,一个线程在run中调用join()方法时,会阻塞当前线程,而让新加入的线程执行完再执行本线程

3、第三个方法是我自己想出来的,就是使用PriorityQueue优先队列,PriorityQueue有这样一个特征,就是可以让队列中某个优先级最高元素最先出来,所以利用这个特性,可以让我们想要哪个线程先执行就先执行,最后通过线程的State来判断该线程是否执行完毕,再执行下一个线程,这个方法的一个好处是各线程之间不必使用join来执行了,各个线程可以独立出来。

PriorityQueue的用法:

PriorityQueue<Integer> queue = new PriorityQueue<Integer>(11, new MyComparator());
        queue.add(3);
        queue.add(1);
        queue.add(-2);
        queue.add(4);
        queue.add(6);
        queue.add(9);
        queue.add(2);
        queue.offer(8);
        while (!queue.isEmpty()) {
            Log.v("zxy", queue.poll() + "");//9864321-2
        }
public static class MyComparator implements Serializable, Comparator<Integer> {

        @Override
        public int compare(Integer lhs, Integer rhs) {
            int value = lhs > rhs ? 1 : lhs < rhs ? -1 : 0;
            return value;
        }
    }

上面这个示例就是优先队列的基本用法,它可以传入一个我们自定义的比较器,利用这个比较器我们可以自己定义队列中元素的顺序。其中compare()方法中的返回值:

1、>0:代表lhs>rhs

2、<0:代表rhs>lhs

3、==0:代表lhs==rhs

而PriorityQueue中有这么几个方法:

1、queue.poll():表示从队列中取出队首上的元素,且取出后remove掉这个元素

2、queue.peek():表示从队列中取出队首上的元素,且取出后不remove掉这个元素

3、queue.add():表示添加一个元素

4、queue.clear():表示清空队列

5、queue.remove():表示移除某个元素

6、queue.size():表示队列的size

利用优先队列实现多个线程顺序执行

final PriorityQueue<Thread> queue = new PriorityQueue<>(11, new MyThreadComparator());
        queue.add(thread1);
        queue.add(thread2);
        queue.add(thread3);
        queue.add(thread5);
        queue.add(thread7);
        queue.add(thread6);
        queue.add(thread4);
        boolean flag = false;
        Thread mThread = null;
        int i=0;
        int size = queue.size();
        while (!queue.isEmpty()) {
            if(!flag){
                mThread = queue.poll();
                mThread.start();
                flag = true;
                i++;
            }
            if(mThread.getState()== Thread.State.TERMINATED && !queue.isEmpty()){
                if(size==i){
                    break;
                }
                if(size==(i+1)){
                    mThread = queue.peek();
                    mThread.start();
                    i++;
                }else {
                    mThread = queue.poll();
                    mThread.start();
                    i++;
                }
            }
        }
public static class MyThreadComparator implements Serializable, Comparator<Thread> {

        @Override
        public int compare(Thread lhs, Thread rhs) {
            int value = lhs.getPriority() < rhs.getPriority() ? 1 : lhs.getPriority() > rhs.getPriority() ? -1 : 0;
            return value;
        }
    }

这里每创建一个线程我都给它设置了一个自带的优先级(其实这个只是一个标志,你也可以用任何东西作为标志,比如采用),用来表示该线程的优先级:

Thread thread1 = new Thread( new Runnable() {
            @Override
            public void run() {
                try {
                    Thread.sleep(4000);
                } catch (InterruptedException e) {
                    e.printStackTrace();
                }
                Log.v("zxy", "result—>thread1");
            }
        });
        thread1.setPriority(3);

利用它做的一些效果:





Android中让多个线程顺序执行探究的更多相关文章

  1. Java中如何保证线程顺序执行

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

  2. T1,T2,T3 三个线程顺序执行

    T1,T2,T3 三个线程顺序执行 现在有 T1.T2.T3 三个线程,你怎样保证 T2 在 T1 执行完后执行,T3 在 T2 执行完后执行?(T1->T2->T3) 这个线程问题通常会 ...

  3. c#线程顺序执行

    using System;using System.Collections.Generic;using System.Linq;using System.Text;using System.Threa ...

  4. Java中线程顺序执行

    现有线程threadone.threadtwo和threadthree,想要的运行顺序为threadone->threadtwo->threadthree,应该如何处理?这里需要用到一个简 ...

  5. js中一个标签在按顺序执行没有被读取到时可以用window.onload

    <%@LANGUAGE="JAVASCRIPT" CODEPAGE="65001"%> <!DOCTYPE html PUBLIC " ...

  6. java中怎么终止一个线程的执行----个人学习心得

    参考了一下两个网站的介绍: ①:http://blog.csdn.net/liuhanhan512/article/details/7077601 ②:http://www.blogjava.net/ ...

  7. Android 中三种启用线程的方法

    在多线程编程这块,我们经常要使用Handler(处理),Thread(线程)和Runnable这三个类,那么他们之间的关系你是否弄清楚了呢? 首先说明Android的CPU分配的最小单元是线程,Han ...

  8. Android中的进程与线程

    四大组件都是运行在主线程中 1.前台进程:用户正在交互,相当于Activity执行了onResume方法 2.可见进程:用户失去了焦点,相当于Activity执行了onPause方法 3.服务进程:运 ...

  9. Android中不能在子线程中更新View视图的原因

    这是一条规律,很多coder知道,但原因是什么呢? 如下: When a process is created for your application, its main thread is ded ...

随机推荐

  1. hive指定hadoop执行队列

    指定队列的命令: 有三种: set mapred.job.queue.name=queue3; SET mapreduce.job.queuename=queue3; set mapred.queue ...

  2. android 自定义view之选座功能

    效果图: 界面比较粗糙,主要看原理. 这个界面主要包括以下几部分 1.座位 2.左边的排数 3.左上方的缩略图 4.缩略图中的红色区域 5.手指移动时跟随移动 6.两个手指缩放时跟随缩放 主要技术点 ...

  3. Mongo DB 初识

    前言 2016年伊始,开始研究NoSql.看了couchdb,cloudant,cassandra,redis.却一直没有看过排行榜第一的mongo,实属不该.近期会花时间研究下mongo.本文是初识 ...

  4. [CSDN_Markdown]Markdown基本语法2

    简介 前文 Markdown基本语法 中介绍了Markdown的基本语法,知道那些基本的语法,实际上已经可以写纯文本的博客了.对我们这群写代码的人或者将要写代码的人来说,貌似这些还不够,我们还希望能插 ...

  5. Unity UGUI实现分段式血条

    我们可以看到像英雄联盟等游戏里英雄头顶的血条显示并非是纯色的,而是根据血量的多少而显示一定量的格子,这种方式明显是比较友好.比较美观的,事实上我们的游戏里面也想实现这样的效果,那该怎么办呢?根据血量的 ...

  6. Android Stutio中使用java8的Lambda表达式

    转载请标明出处: http://blog.csdn.net/xmxkf/article/details/51532028 本文出自:[openXu的博客] 目录: 为什么要使用Lambda表达式 让A ...

  7. 悲观的并发策略——Synchronized互斥锁

    volatile既然不足以保证数据同步,那么就必须要引入锁来确保.互斥锁是最常见的同步手段,在并发过程中,当多条线程对同一个共享数据竞争时,它保证共享数据同一时刻只能被一条线程使用,其他线程只有等到锁 ...

  8. linux中exec和xargs命令的区别和优劣分析

    find的exec及ok命令 exec命令的格式为: exec command {} \; exec后面跟着的是操作命令,然后跟着{}表示每一个参数,然后空格,然后"\;".{}之 ...

  9. MAMP显示文件列表

    背景 MAMP是Mac下的一个PHP+Nginx+MySQL的集成环境,支持多站点,不同版本PHP. 今天有人请教MAMP如何显示文件列表的问题,这里记录一下. 知识补充 一个网站为了安全考虑,默认是 ...

  10. javascript之Style对象

    Background 属性 属性                                 描述 background                      在一行中设置所有的背景属性 ba ...