退出没有循环的线程

时间:2011-04-11 14:49:12

标签: c# multithreading concurrency thread-safety io-completion-ports

我需要一种方法来阻止不包含循环的工作线程。应用程序启动线程,然后线程创建 FileSystemWatcher 对象和 Timer 对象。每个都有回调函数。 到目前为止我所做的是在线程类中添加一个 volatile bool 成员,并使用计时器来检查这个值。一旦设置了这个值,我就会挂掉如何退出线程。

    protected override void OnStart(string[] args)
    {
      try
      {
        Watcher NewWatcher = new Watcher(...);
        Thread WatcherThread = new Thread(NewWatcher.Watcher.Start);
        WatcherThread.Start();
      }
      catch (Exception Ex)
      {
          ...
      }
    }

public class Watcher
{
    private volatile bool _StopThread;

    public Watcher(string filePath)
    {
        this._FilePath = filePath;
        this._LastException = null;

        _StopThread = false;
        TimerCallback timerFunc = new TimerCallback(OnThreadTimer);
        _ThreadTimer = new Timer(timerFunc, null, 5000, 1000);
    }   

    public void Start()
    {
        this.CreateFileWatch();            
    }

    public void Stop()
    {
        _StopThread = true;
    }

    private void CreateFileWatch()
    {
        try
        {
            this._FileWatcher = new FileSystemWatcher();
            this._FileWatcher.Path = Path.GetDirectoryName(FilePath);
            this._FileWatcher.Filter = Path.GetFileName(FilePath);
            this._FileWatcher.IncludeSubdirectories = false;
            this._FileWatcher.NotifyFilter = NotifyFilters.LastWrite;
            this._FileWatcher.Changed += new FileSystemEventHandler(OnFileChanged);

            ...

            this._FileWatcher.EnableRaisingEvents = true;
        }
        catch (Exception ex)
        {
            ...
        }
    }

    private void OnThreadTimer(object source)
    {
        if (_StopThread)
        {
            _ThreadTimer.Dispose();
            _FileWatcher.Dispose();
            // Exit Thread Here (?)
        }
    }

    ...
}

所以当线程被告知停止时我可以处理Timer / FileWatcher - 但是我如何实际退出/停止线程?

6 个答案:

答案 0 :(得分:2)

Start方法退出时,线程将退出。当你到达计时器时,它已经消失了。

答案 1 :(得分:2)

我建议使用ManualResetEvent而不是布尔标志。线程启动FileSystemWatcher,然后等待事件。调用Stop时,会设置事件:

private ManualResetEvent ThreadExitEvent = new ManualResetEvent(false);

public void Start()
{
    // set up the watcher
    this.CreateFileWatch();

    // then wait for the exit event ...
    ThreadExitEvent.WaitOne();

    // Now tear down the watcher and exit.
    // ...
}

public void Stop()
{
    ThreadExitEvent.Set();
}

这可以防止您必须使用计时器,并且您仍然可以收到所有通知。

答案 2 :(得分:1)

一般来说,有两种方法可以做到这一点

  • 使用Thread.Abort()中止线程。这在很大程度上被认为是一个非常危险的操作,因为它将有效地抛出异常并使用它来退出线程。如果代码没有得到充分的准备,这很容易导致泄漏的资源或永远锁定的互斥锁。我会避免这种做法
  • 检查指定位置的_StopThread值,并退出方法或抛出异常以返回到线程开始并从那里优雅地退出线程。这是TPL代码库(取消令牌)
  • 所青睐的方法

答案 3 :(得分:1)

在您的情况下,线程不会立即执行任何操作。您也可以在主线程中创建Watcher对象。

即使在创建线程终止后,Watcher仍然存在。要摆脱它,请使用Dispose。

或者,更具体地说,为什么你甚至想要使用一个线程?要处理来自主线程的不同线程上的事件?在这种情况下,在Watcher事件处理程序中创建一个新线程。

必须小心这一点,以避免过多的线程创建。典型的解决方案是使用线程池/后台工作程序。

答案 4 :(得分:0)

不要使用Thread.Start!

使用TPL,Rx,BackgroundWorker或更高级别的东西。

答案 5 :(得分:0)

http://msdn.microsoft.com/en-us/library/dd997423.aspx

  

您无法取消FromAsync任务,因为基础.NET Framework API当前不支持正在进行的文件或网络I / O取消。您可以将取消功能添加到封装FromAsync调用的方法中,但是您只能在调用FromAsync之前或完成之后响应取消(例如,在延续任务中)。