Java线程详解(7)-线程的调度
Java线程:线程的调度-休眠
Java线程调度是Java多线程的核心,只有良好的调度,才能充分发挥系统的性能,提高程序的执行效率。
这里要明确的一点,不管程序员怎么编写调度,只能最大限度的影响线程执行的次序,而不能做到精准控制。
线程休眠的目的是使线程让出CPU的最简单的做法之一,线程休眠时候,会将CPU资源交给其他线程,以便能轮换执行,当休眠一定时间后,线程会苏醒,进入准备状态等待执行。
线程休眠的方法是Thread.sleep(long millis)和Thread.sleep(long millis, int nanos),均为静态方法,那调用sleep休眠的哪个线程呢?简单说,哪个线程调用sleep,就休眠哪个线程。
/** * Java线程:线程的调度-休眠 */
public class TestSleep { public static void main(String[] args) { Thread t1=new MyThread1(); Thread t2=new Thread(new MyRunnable()); t1.start(); t2.start(); }
}
class MyThread1 extends Thread{ @Override public void run() { for(int i=0;i<3;i++){ System.out.println("线程1第"+i+"次执行!"); try { Thread.sleep(50); } catch (InterruptedException e) { e.printStackTrace(); } } }
}
class MyRunnable implements Runnable{ @Override public void run() { for(int i=0;i<3;i++){ System.out.println("线程2第"+i+"次执行!"); try { Thread.sleep(50); } catch (InterruptedException e) { e.printStackTrace(); } } }
}
执行结果:
线程1第0次执行!
线程2第0次执行!
线程2第1次执行!
线程1第1次执行!
线程2第2次执行!
线程1第2次执行!
从上面的结果输出可以看出,无法精准保证线程执行次序。
Java线程:线程的调度-优先级
与线程休眠类似,线程的优先级仍然无法保障线程的执行次序。只不过,优先级高的线程获取CPU资源的概率较大,优先级低的并非没机会执行。
线程的优先级用1-10之间的整数表示,数值越大优先级越高,默认的优先级为5。
在一个线程中开启另外一个新线程,则新开线程称为该线程的子线程,子线程初始优先级与父线程相同。
/** * Java线程:线程的调度-优先级 */
public class TestPriority { public static void main(String[] args) { Thread t1=new MyThread1(); Thread t2=new Thread(new MyRunnable()); t1.setPriority(10); t2.setPriority(1); t1.start(); t2.start(); }
}
class MyThread1 extends Thread{ @Override public void run() { for(int i=0;i<10;i++){ System.out.println("线程1第"+i+"次执行!"); try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } }
}
class MyRunnable implements Runnable{ @Override public void run() { for(int i=0;i<10;i++){ System.out.println("线程2第"+i+"次执行!"); try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); } } }
}
执行结果:
线程1第0次执行!
线程1第1次执行!
线程1第2次执行!
线程2第0次执行!
线程1第3次执行!
线程2第1次执行!
线程1第4次执行!
线程2第2次执行!
线程1第5次执行!
线程2第3次执行!
线程1第6次执行!
线程2第4次执行!
线程1第7次执行!
线程2第5次执行!
线程1第8次执行!
线程2第6次执行!
线程1第9次执行!
线程2第7次执行!
线程2第8次执行!
线程2第9次执行!
Java线程:线程的调度-让步
线程的让步含义就是使当前运行着线程让出CPU资源,但是让给谁不知道,仅仅是让出,线程状态回到可运行状态。
线程的让步使用Thread.yield()方法,yield()为静态方法,功能是暂停当前正在执行的线程对象,并执行其他线程。
/** * Java线程:线程的调度-让步 */
public class Test { public static void main(String[] args) { Thread t1=new MyThread1(); Thread t2=new Thread(new MyRunnable()); t1.start(); t2.start(); }
}
class MyThread1 extends Thread{ @Override public void run() { for(int i=0;i<10;i++){ System.out.println("线程1第"+i+"次执行!"); } }
}
class MyRunnable implements Runnable{ @Override public void run() { for(int i=0;i<10;i++){ System.out.println("线程2第"+i+"次执行!"); Thread.yield(); } }
}
执行结果:
线程2第0次执行!
线程1第0次执行!
线程1第1次执行!
线程1第2次执行!
线程1第3次执行!
线程1第4次执行!
线程1第5次执行!
线程1第6次执行!
线程1第7次执行!
线程1第8次执行!
线程1第9次执行!
线程2第1次执行!
线程2第2次执行!
线程2第3次执行!
线程2第4次执行!
线程2第5次执行!
线程2第6次执行!
线程2第7次执行!
线程2第8次执行!
线程2第9次执行!
Java线程:线程的调度-合并
线程的合并的含义就是将几个并行线程的线程合并为一个单线程执行,应用场景是当一个线程必须等待另一个线程执行完毕才能执行时可以使用join方法。
join为非静态方法,定义如下:
void join()——等待该线程终止。
void join(longmillis)——等待该线程终止的时间最长为 millis毫秒。
void join(longmillis,int nanos)——等待该线程终止的时间最长为 millis毫秒 + nanos 纳秒。
/** * Java线程:线程的调度-合并 */
public class Test { public static void main(String[] args) { Thread t1=new MyThread1(); t1.start(); for (int i = 0; i < 20; i++) { System.out.println("主线程第" + i +"次执行!"); if (i>2) { try { ///t1线程合并到主线程中,主线程停止执行过程,转而执行t1线程,直到t1执行完毕后继续。 t1.join(); } catch (InterruptedException e) { e.printStackTrace(); } } } }
}
class MyThread1 extends Thread{ @Override public void run() { for(int i=0;i<10;i++){ System.out.println("线程1第"+i+"次执行!"); } }
}
执行结果:
主线程第0次执行!
主线程第1次执行!
主线程第2次执行!
主线程第3次执行!
线程1第0次执行!
线程1第1次执行!
线程1第2次执行!
线程1第3次执行!
线程1第4次执行!
线程1第5次执行!
线程1第6次执行!
线程1第7次执行!
线程1第8次执行!
线程1第9次执行!
主线程第4次执行!
主线程第5次执行!
主线程第6次执行!
主线程第7次执行!
主线程第8次执行!
主线程第9次执行!
主线程第10次执行!
主线程第11次执行!
主线程第12次执行!
主线程第13次执行!
主线程第14次执行!
主线程第15次执行!
主线程第16次执行!
主线程第17次执行!
主线程第18次执行!
主线程第19次执行!
Java线程:线程的调度-守护线程
守护线程与普通线程写法上基本么啥区别,调用线程对象的方法setDaemon(true),则可以将其设置为守护线程。
守护线程使用的情况较少,但并非无用,举例来说,JVM的垃圾回收、内存管理等线程都是守护线程。还有就是在做数据库应用时候,使用的数据库连接池,连接池本身也包含着很多后台线程,监控连接个数、超时时间、状态等等。
setDaemon方法的详细说明:
public final void setDaemon(boolean on)将该线程标记为守护线程或用户线程。当正在运行的线程都是守护线程时,Java虚拟机退出。
该方法必须在启动线程前调用。
该方法首先调用该线程的 checkAccess方法,且不带任何参数。这可能抛出 SecurityException(在当前线程中)。
参数:on - 如果为true,则将该线程标记为守护线程。
抛出:
IllegalThreadStateException- 如果该线程处于活动状态。
SecurityException- 如果当前线程无法修改该线程。
另请参见:
isDaemon(),checkAccess()
/** * Java线程:线程的调度-守护线程 */
public class Test { public static void main(String[] args) { Thread t1=new MyCommon(); Thread t2=new Thread(new MyDaemon()); t2.setDaemon(true);//设置为守护线程 t2.start(); t1.start(); }
}
class MyCommon extends Thread{ @Override public void run() { for(int i=0;i<5;i++){ System.out.println("线程1第"+i+"次执行!"); try { Thread.sleep(7); } catch (InterruptedException e) { e.printStackTrace(); } } }
}
class MyDaemon implements Runnable{ @Override public void run() { for (long i = 0; i < 9999999L; i++) { System.out.println("后台线程第" + i +"次执行!"); try { Thread.sleep(7); } catch (InterruptedException e) { e.printStackTrace(); } } }
}
执行结果:
线程1第0次执行!
后台线程第0次执行!
后台线程第1次执行!
线程1第1次执行!
后台线程第2次执行!
线程1第2次执行!
后台线程第3次执行!
线程1第3次执行!
后台线程第4次执行!
线程1第4次执行!
后台线程第5次执行!
后台线程第6次执行!
后台线程第7次执行!
后台线程第8次执行!
后台线程第9次执行!
后台线程第10次执行!
从上面的执行结果可以看出:
前台线程是保证执行完毕的,后台线程还没有执行完毕就退出了。
实际上:JRE判断程序是否执行结束的标准是所有的前台执线程行完毕了,而不管后台线程的状态,因此,在使用后台县城时候一定要注意这个问题。
Java线程详解(7)-线程的调度相关推荐
- Java多线程详解(线程不安全案例)
嗨喽-小伙伴们我又来了, 通过前面两章的学习,我们了解了线程的基本概念和创建线程的四种方式. 附上链接: 1. Java多线程详解(基本概念) 2. Java多线程详解(如何创建线程) ...
- java executors 详解_线程池—Executors 详解
各位志同道合的朋友们大家好,我是一个一直在一线互联网踩坑十余年的编码爱好者,现在将我们的各种经验以及架构实战分享出来,如果大家喜欢,就关注我,一起将技术学深学透,我会每一篇分享结束都会预告下一专题 线 ...
- Java线程详解(11)-线程池
Sun在Java5中,对Java线程的类库做了大量的扩展,其中线程池就是Java5的新特征之一,除了线程池之外,还有很多多线程相关的内容,为多线程的编程带来了极大便利.为了编写高效稳定可靠的多线程程序 ...
- Java线程详解(8)-线程的同步
Java线程:线程的同步-同步方法 线程的同步是保证多线程安全访问竞争资源的一种手段. 线程的同步是Java多线程编程的难点,往往开发者搞不清楚什么是竞争资源.什么时候需要考虑同步,怎么同步等等问题, ...
- java线程详解_Java线程详解
程序.进程.线程的概念程序(program):是为完成特定任务.用某种语言编写的一组指令的集合.即指一段静态的代码,静态对象. 进程(process):是程序的一次执行过程,或是正在运行的一个程序.动 ...
- Java多线程详解(线程池)
嗨喽-小伙伴们我来了, 上一章我们介绍了Java中的Thread类里一些常用的方法.本节我们就来聊一聊线程池. 说到"池",大家或许都不陌生,在java中,我们有见过数据库连接池, ...
- java多线程详解及线程池创建
多线程 线程 线程是独立的执行路径 在程序执行时,即使自己没有创建线程,后台也会有多个线程 main()称为主线程,为系统的入口,用于执行整个程序 在一个进程中如果开辟了多个线程,线程的运行由调度器安 ...
- java executors 详解_线程池Executors详解
为什么要用线程池呢? 一是减少了创建和销毁线程的次数,每个工作线程都可以被重复利用,可执行多个任务; 二是可以根据系统的承受能力,调整线程池中工作线线程的数目,防止因为因为消耗过多的内存,而把服务器累 ...
- Java线程详解(6)-线程的交互
线程交互是比较复杂的问题,SCJP要求不很基础:给定一个场景,编写代码来恰当使用等待.通知和通知所有线程. 一.线程交互的基础知识 SCJP所要求的线程交互知识点需要从java.lang.Object ...
最新文章
- 【直播回顾及资料下载】小程序云应用入门实操系列课程第一讲...
- altium designer布线调整时不自动删除旧线
- ceil和floor
- Android---- 获取当前应用的版本号和当前android系统的版本号
- Linux 命令(15)—— umask 命令(builtin)
- 德州扑克AI--Programming Poker AI(译)
- 简易安装Matlab R2014a 破解版
- 华为数通HCIA学习笔记之数据通信与网络基础(一)
- python微信api_python调用企业微信API
- 论文阅读-Generative Image Inpainting with Contextual Attention
- 实现了一个跨平台的 YUV 文件图片查看器
- Arduino笔记实验(初级阶段)—DHT11温湿度传感器
- php 中国地图编辑,科学网—如何正确绘制中国示意性地图?编辑给您提建议 - 科学出版社的博文...
- *CF1132D.Stressful Training(二分+队列+贪心)
- matlab测量直流母线上的电压,直流母线
- visio2019画思维导图
- Facebook广告效果数据获取
- 抓取新浪微博好友昵称和性别
- 黎曼的猜想 MySQL案例练习记录
- 低成本佐川泽川圣安变频器设计方案
热门文章
- OpenShift Rancher 进行持续集成
- Apache Beam欲通过uber api获取大数据
- 一种创建进程间COM来启动IE的方式
- 世博展示生态化住宅 物联网助推智能化家居
- shell中if条件之数值的比较方法
- 【opencv+机器学习】error C3083: 'ml': the symbol to the left of a '::' must be a type问题原因
- 概率整形在比特度量判决(BMD)下的可达速率计算
- Nginx之负载均衡
- Hyperledger Fabric权限进阶篇
- linux shell脚本学习xargs命令使用详解