我需要一种方法来阻止不包含循环的工作线程。应用程序启动线程,然后线程创建 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 - 但是我如何实际退出/停止线程?
答案 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之前或完成之后响应取消(例如,在延续任务中)。