本文主要是介绍线程学习笔记(五)-线程让步和线程守护,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!
在《Java范例大全》这本书上,线程让步和线程守护是分为两个例子来讲的,但是线程让步和线程守护都在讲一个等字,我认为它们两个可以合到一个程序里面来看现象。另外这本书上关于守护线程的例子有问题。
(1) 线程让步(yield)让当前运行的线程放弃其所占用的CPU时间片,以便让其他线程运行,用线程让步的目的是让线程适当地轮转,但并不能保证达到此效果。因为即使当前线程放弃时间片,还有可能再次被Java虚拟机(JVM)选中。
(2)线程执行一次让步操作只是让步一个时间片,当下次时间片轮到该线程时,该线程会继续运行。由于一个时间片非常短,最多几个毫秒,因此一次线程让步操作微乎其微,甚至觉察不出,但是我们可以在程序中执行输出语句来查看线程之间时间片的轮转。
(3)线程让步适用于 同级别的线程让步,且让步后不是转入阻塞状态,则是就绪状态,不抛出异常。如果不是不同级别的,高级别的让出时间片,还是它再获取时间片。
(4)Java虚拟机(JVM)的垃圾回收线程可以称为守护线程。 守护线程是一种特殊的线程,它是否运行结束并不仅仅依赖于自己run()方法内的程序,还依赖于其他非守护线程是否已经运行结束。普通线程可以通过Thread类的setDaemon方法设置为守护线程,参数为true时表示该线程为守护线程。任何线程都可以通过Thread类的isDaemon()方法判断是否为守护线程。
(5)如果线程已开始运行,Thread类的setDaemon方法无效,即必须在调用方法start()之前调用setDaemon方法,才能设置该线程为守护线程。
(6)如果所有的非守护线程运行结束,无论守护线程有没有运行结束,都将不再运行。如果一个程序没有任何守护线程,则所有非守护线程运行结束后就退出。程序中启动的线程默认为非守护线程,但在守护线程中启动的线程都是守护线程。
(7)本例子利用两个线程用线程让步来实现模拟传文件。
package core;import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Vector;public class TextThreadYield {private Vector<String> vector = new Vector<String>();// 创建向量集合private DateFormat dateFormat = new SimpleDateFormat("HH-mm-ss:SSSS");private boolean isFlag = false;private class Yield extends Thread {// 让步接收文件夹public Yield() {this.setName("接收文件");// 设置线程名称this.setDaemon(true);// 如果SendFile线程结束,则该线程自动结束}public void run() {while (!isFlag) {// 标识为真进行循环try {Thread.sleep(100);} catch (InterruptedException e) {System.out.println("唤醒异常:" + e.getMessage());}if (vector.size() == 0) {System.out.println(dateFormat.format(new Date()) + "\t向量集合中没有文件,执行yield时间陪让步操作");Thread.yield();// 调用线程让步} else {String ss=(String)vector.remove(0);//移除文件获得对象System.out.println(dateFormat.format(new Date())+"\t取到文件,名为"+ss);}}}}private class SendFile extends Thread{//发送文件private String[] files=new String[]{"新闻文件","国内旅游向导","山水名画欣赏","发家致富说明"};public SendFile(){this.setName("发送文件");}public void run(){try {for (int i = 0; i < files.length; i++) {Thread.sleep(200);//线程休眠vector.add(files[i]);//向vector中添加文件System.out.println(dateFormat.format(new Date()) + "\t把文件"+files[i]+"发送到vector中");}Thread.sleep(100); //线程休眠} catch (Exception e) {System.out.println("唤醒异常:" + e.getMessage());}}}public static void main(String[] args) {TextThreadYield text = new TextThreadYield();text.new Yield().start();// 启动接收文件的线程text.new SendFile().start();// 启动发送文件的线程}
}
源程序解读
源程序稍微做了些修改,使得更容易理解
(1)TextThreadYield 定义两个私有内部类作为线程A-接收文件,线程B-发送文件。
这两个内部类的调用都是通过 主类名.new 内部类名(参数列表);的方式进行实例化。
(2)TextThreadYield类创建向量集合操作传送的文件,声明一标识变量控制循环,这一标识变量在此程序中设置为固定值,如果程序较为复杂,它的值可能会变化,来控制线程的运行。
(3)对于让步接收文件类Yield的构造方法设置线程的名称并设置其为守护线程,如果发送线程不运行,其也不会运行,发送线程运行结束,它也会运行结束。
(4)SendFile类,在其run()方法中利用循环使得线程休眠并向向量集合中添加文件元素。设置线程的休眠时间以使Yield类线程就行时间让步。
一般而言,线程的yield操作效果是非常小的,将Thread.yield()一行注释掉,再次运行与之前的结果比较,发现两者的运行情况是一样的。Thread.yield()的具体应用场景自此就迷茫了。
源程序稍微做了些修改,使得更容易理解
(1)TextThreadYield 定义两个私有内部类作为线程A-接收文件,线程B-发送文件。
这两个内部类的调用都是通过 主类名.new 内部类名(参数列表);的方式进行实例化。
(2)TextThreadYield类创建向量集合操作传送的文件,声明一标识变量控制循环,这一标识变量在此程序中设置为固定值,如果程序较为复杂,它的值可能会变化,来控制线程的运行。
(3)对于让步接收文件类Yield的构造方法设置线程的名称并设置其为守护线程,如果发送线程不运行,其也不会运行,发送线程运行结束,它也会运行结束。
(4)SendFile类,在其run()方法中利用循环使得线程休眠并向向量集合中添加文件元素。设置线程的休眠时间以使Yield类线程就行时间让步。
一般而言,线程的yield操作效果是非常小的,将Thread.yield()一行注释掉,再次运行与之前的结果比较,发现两者的运行情况是一样的。Thread.yield()的具体应用场景自此就迷茫了。
这篇关于线程学习笔记(五)-线程让步和线程守护的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!