线程学习笔记(五)-线程让步和线程守护

本文介绍Java中线程让步(yield)与守护线程的基本概念及使用方法,并通过具体示例演示如何利用这两种机制实现线程间的交互。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在《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()的具体应用场景自此就迷茫了。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值