`

Two Phase Termination

 
阅读更多

        在主线程中启动一个线程t,当主线程向t发出停止执行,t用一个boolean值来控制run中的不断执行过程。t进行终止操作,这里的终止不是是使得线程执行完run方法。

public class CountupThread extends Thread {
    private long counter=0;  //计数器
    private volatile boolean shutdownRequested=false;
    
    public void shutdownRequest(){
    	shutdownRequested=true;
    	interrupt();
    }
    public boolean isShutdownRequested(){
    	return shutdownRequested;
    }
    public final void run(){
    	try{
    		while(!shutdownRequested){
    		    doWork();
    		}
    	}catch(InterruptedException e){
    		//真在sleep的时候interrupted
    	}finally{
    		System.out.println("doWorkShutdown:counter="+counter);
    	}
    }
	private void doWork() throws InterruptedException {
		counter++;
		System.out.println("do Worker counter:"+counter);
		Thread.sleep(500);
	}
}

 

public class Main {
	public static void main(String[] args){
		System.out.println("main:Begin");
		try{
			CountupThread t=new CountupThread();
			t.start();
			Thread.sleep(10000);
			System.out.println("main:shutDownRequest");
			t.shutdownRequest();
			System.out.println("main:join");
			t.join();//等待线程结束,可以用isAlive来检测是否结束
		}catch(InterruptedException e){
			e.printStackTrace();
		}
		System.out.println("main:End");
	}	
}

 运行结果:

main:Begin
do Worker counter:1
do Worker counter:2
do Worker counter:3
do Worker counter:4
do Worker counter:5
do Worker counter:6
do Worker counter:7
do Worker counter:8
do Worker counter:9
do Worker counter:10
do Worker counter:11
do Worker counter:12
do Worker counter:13
do Worker counter:14
do Worker counter:15
do Worker counter:16
do Worker counter:17
do Worker counter:18
do Worker counter:19
do Worker counter:20
main:shutDownRequest
main:join
doWorkShutdown:counter=20
main:End

这里,考虑到了:

        安全性:当t收到结束信号以后并没有马上结束,而是先改变结束标识shutdownRequest。这时对象不会被突然地终止而受到破坏。

        生命性:改变结束标识后调用了interrupt方法,这主要是为了使得当线程正在sleep、wait的时候也能够顺利中断掉。为了使抛出异常的时候也能终止处理故使用了try...finally块。shutdownRequest的作用是如果任务很繁重,它会执行到下次循环的。

        响应性:终止请求发出后要尽快进入终止处理。

 

关于interrupt的一点说明

        当一个线程调用interrupt方法后,线程会进入两种状态:一、线程中断状态改变;二、线程没有中断,而是进入InterruptedException(通常是线程处于sleep、wait、join)。

       

public class Thread1 extends Thread {
	@Override
	public void run(){
		try {
			System.out.println("1");
			System.out.println("1");
			sleep(5000);  //休息5秒吧
			System.out.println("2");
			System.out.println("2");
		} catch (InterruptedException e) {
			//如果sleep时进入异常,那么上面的2就没有输出
			e.printStackTrace(); //这时线程没有中断,会继续输出下面的3
		}
		System.out.println("3");
		System.out.println("3");
	}

 

public class Test {
    public static void main(String[] args){
    	Thread t1=new Thread1();
    	t1.start();
    	t1.interrupt();
    }
}

 运行结果:

1
java.lang.InterruptedException: sleep interrupted
    at java.lang.Thread.sleep(Native Method)
1
3
3
    at interrupt.Thread1.run(Thread1.java:9)

 

如果要让线程由InterruptedException进入中断,应:

try {
    ... ...
} catch (InterruptedException e) {
    Thread.currentThread().interrupt();
}

 

        对于第一种情况:Thread.interrupt()方法不会中断一个正在运行的线程。它只是发去一个信号。

public class Thread2 extends Thread{
	@Override
	public void run(){
		int i=0;
		while(true)
	        	System.out.println(i++);  
     }
}

 

public class Test {
    public static void main(String[] args) throws InterruptedException{
    	Thread t2=new Thread2();
    	t2.start();
    	Thread.sleep(100);
    	t2.interrupt();
    }
}

        运行结果中可以知道,线程在一直运行,根本没有停止!

        应该Thread.interrupted()检查是否发生中断。Thread.interrupted()能告诉你线程是否发生中断,并将清除中断状态标记,所以程序不会两次通知你线程发生了中断。

      

public class Thread2 extends Thread{
	@Override
	public void run(){
		int i=0;
		//while(true)
		//	System.out.println(i++);  
		while(!Thread.interrupted())
			System.out.println(i++); 
     }
}

 

分享到:
评论

相关推荐

    设计模式之-两阶段终止模式(Two-Phase Termination Patter)模式.docx

    两阶段终止模式(Two-Phase Termination Pattern)是一种在多线程编程中确保线程安全关闭的方法,由黄文海在其著作《Java多线程编程实战指南 设计模式》中提出,不属于传统的23种设计模式,而是扩展到36种设计模式之...

    汪文君高并发编程实战视频资源全集

    │ 高并发编程第二阶段35讲、多线程Two Phase Termination设计模式-下.mp4 │ 高并发编程第二阶段36讲、多线程Worker-Thread设计模式-上.mp4 │ 高并发编程第二阶段37讲、多线程Worker-Thread设计模式-上.mp4 │...

    汪文君高并发编程实战视频资源下载.txt

    │ 高并发编程第二阶段35讲、多线程Two Phase Termination设计模式-下.mp4 │ 高并发编程第二阶段36讲、多线程Worker-Thread设计模式-上.mp4 │ 高并发编程第二阶段37讲、多线程Worker-Thread设计模式-上.mp4 │...

    Java多线程详解

    Java多线程模式详解 ...10、Two-Phase Termination ———— 快把玩具收拾好,去睡觉吧 11、Thread-Specific Storage ———— 每个线程的保管箱 12、Active Object ———— 接受异步消息的主动对象

    java多线程设计模式 (PDF中文版, 附源码)

    第10章 Two-Phase Termination——快把玩具收拾好,去睡觉吧 第11章 Thread-Specific Storage——每个线程的保管箱 第12章 Active Object——接受异步消息的主动对象 总结 多线程程序设计的模式语言 附录A 练习问题...

    JavaConcurrencyPattern:Java并发模式

    Two-phase Termination模式: 【前言】 随着现代CPU的生产Craft.io从提升CPU主频频率转向多核化,即在一块芯片上集成多个CPU内核(Core),以往那种靠CPU自身处理能力的提升所带来的软件计算性能提升的那种“免费...

    java多线程设计模式详解(PDF及源码)

    Phase Termination——快把玩具收拾好,去睡觉吧 第11章 Thread-Specific Storage——每个线程的保管箱 第12章 Active Object——接受异步消息的主动对象 总结 多线程程序设计的模式语言 附录A 练习问题的解答 附录B...

    16、常用并发设计模式精讲(1).pdf

    - **两阶段终止模式**(Two-phase Termination Pattern)是一种用于优雅终止线程的设计模式。该模式通过两个步骤来终止线程:第一步是发送终止请求;第二步是等待线程自行终止。 **步骤详解:** 1. **发送终止请求...

    良葛格DesignPattern学习笔记

    - **Two-phase Termination**(两阶段终止模式):允许一个进程安全地终止,即使它拥有多个子线程。 - **Read-Write Lock**(读写锁模式):允许多个读取线程同时访问资源,但只允许一个写入线程访问资源。 - **...

    设计模式笔记

    - **Two-phase Termination(两阶段终止模式)**:优雅地关闭一个服务,避免突然关闭导致数据丢失等问题。 - **Thread-Specific Storage(线程本地存储模式)**:每个线程都有自己的独立副本,可以避免线程间的竞争...

    计算机网络测试题目 英语题目

    - **TCP Connection Termination**:TCP连接的终止分为正常终止和异常终止两种情况。 - **正常终止步骤**: 1. 客户端发送一个FIN段来关闭连接。 2. 服务器接收到FIN后,发送ACK确认。 3. 服务器发送自己的FIN...

    Pattens In Java.pdf

    ### 基本设计模型(Fundamental Design Patterns) #### 1.1 授权(Delegation)(When not to use inheritance) **特点:** 在面向对象编程中,继承是一种常用的设计方式,但有时它并不是最合适的解决方案。...

    中兴通讯专业词汇库(040924)

    13. **(CPFSK)连续相位频移键控 (Continuous Phase Frequency Shift Keying)** - **定义**:一种调制技术,用于数据传输。 - **应用场景**:在无线通信领域,特别是在移动通信系统中。 14. **(CT2无绳电话的)...

Global site tag (gtag.js) - Google Analytics