线程调度是指按照特定机制为多个线程分配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的用法:

  1. PriorityQueue<Integer> queue = new PriorityQueue<Integer>(11, new MyComparator());
  2. queue.add(3);
  3. queue.add(1);
  4. queue.add(-2);
  5. queue.add(4);
  6. queue.add(6);
  7. queue.add(9);
  8. queue.add(2);
  9. queue.offer(8);
  10. while (!queue.isEmpty()) {
  11. Log.v("zxy", queue.poll() + "");//9864321-2
  12. }
  13. public static class MyComparator implements Serializable, Comparator<Integer> {
  14. @Override
  15. public int compare(Integer lhs, Integer rhs) {
  16. int value = lhs > rhs ? 1 : lhs < rhs ? -1 : 0;
  17. return value;
  18. }
  19. }

上面这个示例就是优先队列的基本用法,它可以传入一个我们自定义的比较器,利用这个比较器我们可以自己定义队列中元素的顺序。其中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

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

  1. final PriorityQueue<Thread> queue = new PriorityQueue<>(11, new MyThreadComparator());
  2. queue.add(thread1);
  3. queue.add(thread2);
  4. queue.add(thread3);
  5. queue.add(thread5);
  6. queue.add(thread7);
  7. queue.add(thread6);
  8. queue.add(thread4);
  9. boolean flag = false;
  10. Thread mThread = null;
  11. int i=0;
  12. int size = queue.size();
  13. while (!queue.isEmpty()) {
  14. if(!flag){
  15. mThread = queue.poll();
  16. mThread.start();
  17. flag = true;
  18. i++;
  19. }
  20. if(mThread.getState()== Thread.State.TERMINATED && !queue.isEmpty()){
  21. if(size==i){
  22. break;
  23. }
  24. if(size==(i+1)){
  25. mThread = queue.peek();
  26. mThread.start();
  27. i++;
  28. }else {
  29. mThread = queue.poll();
  30. mThread.start();
  31. i++;
  32. }
  33. }
  34. }
  35. public static class MyThreadComparator implements Serializable, Comparator<Thread> {
  36. @Override
  37. public int compare(Thread lhs, Thread rhs) {
  38. int value = lhs.getPriority() < rhs.getPriority() ? 1 : lhs.getPriority() > rhs.getPriority() ? -1 : 0;
  39. return value;
  40. }
  41. }

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

  1. Thread thread1 = new Thread( new Runnable() {
  2. @Override
  3. public void run() {
  4. try {
  5. Thread.sleep(4000);
  6. } catch (InterruptedException e) {
  7. e.printStackTrace();
  8. }
  9. Log.v("zxy", "result—>thread1");
  10. }
  11. });
  12. 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. how to output quotes in bash prompt

    introduction In certain situations, quotes are required to be output in the command prompt. To do th ...

  2. ThinkPHP 初探

    准备 ThinkPHP下载 Eclipse-for-php 如何使用 放置位置 检验引用效果 效果 路由 调试之模板的使用 前提 生产模式 开发模式 添加完相应的路径以及模板文件后 总结 对国人开发的 ...

  3. Java通过实现Runnable接口来创建线程

    创建一个线程,最简单的方法是创建一个实现Runnable接口的类. 为了实现Runnable,一个类只需要执行一个方法调用run(),声明如下: public void run() 你可以重写该方法, ...

  4. npm管理工具介绍

    概述 Npm是NodeJS包管理工具,在最新版本中Nodejs集成了npm,可以通过输入 "npm -v" 来测试是否成功安装.如果你安装的是旧版本的 npm,可以通过 npm 命 ...

  5. 21 viewPager--- hzScrollView ----llContainer

    结构: MainActivity.java package com.qf.day21_hsviewpagerfragment_demo5; import java.util.ArrayList; im ...

  6. Spring开发环境搭建教程

    Spring开发环境搭建 JDK7以上版本 eclispe for j2ee 4.0以上版本 Spring frameWorks 3.0以上版本 至于前两个我们就不介绍,直接百度就可以了,对于Spri ...

  7. 【Netty源码学习】ChannelPipeline(一)

    ChannelPipeline类似于一个管道,管道中存放的是一系列对读取数据进行业务操作的ChannelHandler. 1.ChannelPipeline的结构图: 在之前的博客[Netty源码学习 ...

  8. OpenCV相机标定

    标签(空格分隔): Opencv 相机标定是图像处理的基础,虽然相机使用的是小孔成像模型,但是由于小孔的透光非常有限,所以需要使用透镜聚焦足够多的光线.在使用的过程中,需要知道相机的焦距.成像中心以及 ...

  9. 【java集合框架源码剖析系列】java源码剖析之HashMap

    前言:之所以打算写java集合框架源码剖析系列博客是因为自己反思了一下阿里内推一面的失败(估计没过,因为写此博客已距阿里巴巴一面一个星期),当时面试完之后感觉自己回答的挺好的,而且据面试官最后说的这几 ...

  10. CONFIGURE ADFS 3.0 WITH SHAREPOINT 2013

     http://blogit.create.pt/miguelmoreno/2014/11/14/configure-adfs-3-0-with-sharepoint-2013/