线程应当如何正常退出?

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了线程应当如何正常退出?相关的知识,希望对你有一定的参考价值。

参考技术A 先来看这样一段代码:
#include <windows.h
#include <cstdioclass Tpublic:~T()puts(&quot;destruction&quot;);;;DWORD WINAPI ThreadProc(LPVOID)T t;ExitThread(0);int main()CreateThread(NULL, 1024, ThreadProc, NULL, 0, NULL);
Sleep(1000);return 0;
这是一个简单的线程调用的例子,为了设置线程的退出码,有的人喜欢调用ExitThread退出线程,但这不是一种好的方式,运行这个程序就会发现,ThreadProc中的局部变量t没有调用析构函数!
很多人没有意识到,ExitThread这个函数有强制关闭线程的意思,强制关闭,就不是正常退出,ExitThread函数后面的代码将不能被执行。而像动态变量的析构函数是在函数花括号结束处被调用的,这些代码是编译器隐式地添加到ExitThread之后的内容。所以,如果直接调用ExitThread,析构函数将被跳过,这在一些程序中将是一个严重的问题。而退出码的设置,也只需要用一个return就行了。上面的代码应该写成这样:
DWORD WINAPI ThreadProc(LPVOID)T t;return 0;类似的函数还有ExitProcess,进程正常退出的情况也不应该使用ExitProcess,原因如上面的ExitThread一样,ExitProcess是强制退出一个进程,析构函数及进程退出时必须的一些运行库清理工作也会因强制退出而不能被执行。本回答被提问者采纳
参考技术B 终止线程的三种方法
1. 使用退出标志,使线程正常退出,也就是当run方法完成后线程终止。
2. 使用stop方法强行终止线程(这个方法不推荐使用,因为stop和suspend、resume一样,也可能发生不可预料的结果)。
3. 使用interrupt方法中断线程。
使用退出标志终止线程 :
当run方法执行完后,线程就会退出。但有时run方法是永远不会结束的。如在服务端程序中使用线程进行监听客户端请求,或是其他的需要循环处理的任务。在这种情况下,一般是将这些任务放在一个循环中,如while循环。如果想让循环永远运行下去,可以使用while(true)……来处理。但要想使while循环在某一特定条件下退出,最直接的方法就是设一个boolean类型的标志,并通过设置这个标志为true或false来控制while循环是否退出。下面给出了一个利用退出标志终止线程的例子。
package chapter2;
public class ThreadFlag extends Thread

public volatile boolean exit = false;
public void run()

while (!exit);

public static void main(String[] args) throws Exception

ThreadFlag thread = new ThreadFlag();
thread.start();
sleep(5000); // 主线程延迟5秒
thread.exit = true; // 终止线程thread
thread.join();
System.out.println("线程退出!");


在上面代码中定义了一个退出标志exit,当exit为true时,while循环退出,exit的默认值为false.在定义exit时,使用了一个Java关键字volatile,这个关键字的目的是使exit同步,也就是说在同一时刻只能由一个线程来修改exit的值,

2. 使用stop方法终止线程
使用stop方法可以强行终止正在运行或挂起的线程。我们可以使用如下的代码来终止线程:
thread.stop();
虽然使用上面的代码可以终止线程,但使用stop方法是很危险的,就象突然关闭计算机电源,而不是按正常程序关机一样,可能会产生不可预料的结果,因此,并不推荐使用stop方法来终止线程。

3. 使用interrupt方法终止线程
使用interrupt方法来终端线程可分为两种情况:
(1)线程处于阻塞状态,如使用了sleep方法。
(2)使用while(!isInterrupted())……来判断线程是否被中断。
在第一种情况下使用interrupt方法,sleep方法将抛出一个InterruptedException例外,而在第二种情况下线程将直接退出。下面的代码演示了在第一种情况下使用interrupt方法。
package chapter2;
public class ThreadInterrupt extends Thread

public void run()

try

sleep(50000); // 延迟50秒

catch (InterruptedException e)

System.out.println(e.getMessage());


public static void main(String[] args) throws Exception

Thread thread = new ThreadInterrupt();
thread.start();
System.out.println("在50秒之内按任意键中断线程!");
System.in.read();
thread.interrupt();
thread.join();
System.out.println("线程已经退出!");


上面代码的运行结果如下:
在50秒之内按任意键中断线程!
sleep interrupted
线程已经退出!
在调用interrupt方法后, sleep方法抛出异常,然后输出错误信息:sleep interrupted.
注意:在Thread类中有两个方法可以判断线程是否通过interrupt方法被终止。一个是静态的方法interrupted(),一个是非静态的方法isInterrupted(),这两个方法的区别是interrupted用来判断当前线是否被中断,而isInterrupted可以用来判断其他线程是否被中断。因此,while (!isInterrupted())也可以换成while (!Thread.interrupted())。

如何退出 NSThread

【中文标题】如何退出 NSThread【英文标题】:How to exit NSThread 【发布时间】:2009-05-26 07:02:39 【问题描述】:

我正在使用这样的线程,

[NSThread detachNewThreadSelector:@selector(myfunction) toTarget:self withObject

线程运行正常,我想在中间退出线程,怎么办。如果我使用[NSThread exit]应用程序挂起。

【问题讨论】:

【参考方案1】:

您在哪个线程中运行“[NSThread exit]”? [NSThread exit] 在 current 线程中运行,因此您需要将其作为 myfunction 选择器的一部分来调用。如果你在主线程中调用它,它只会退出主线程。

此外,像这样停止线程也不是一个好主意,因为它会阻止线程退出清理资源。

myfunction 应该基于与协调线程的共享变量退出。

- (void) myFunction

    while([someObject stillWorkToBeDone]) 
     
      performBitsOfWork();
    

您可以使用“withObject”在协调线程和工作线程之间共享一个引用。这样,协调线程可以更改共享对象中的一个实例变量,以便工作线程可以根据这个条件停止它的工作。

要退出工作线程,协调线程只需调用 smth 如下:

[someObject setStillWorkToBeDone:false];

【讨论】:

【参考方案2】:

你应该打电话

-[NSThread cancel]

在你创建的线程上检查

-[NSThread isCancelled]

在你的 while 循环中。

【讨论】:

以上是关于线程应当如何正常退出?的主要内容,如果未能解决你的问题,请参考以下文章

主线程创建了子线程,怎么让主线程退出,而子线程仍然运行

MFC在视类中如何退出程序

linux c开发: 在程序退出时进行处理

什么时候最好在看门狗线程中导致段错误而不是正常退出以停止进程?

如何停止一个进程

Python线程退出代码