1

首先,代码:

lblFileNbr.Text = "?/?";
lblFileNbr.ToolTipText = "Searching for files...";
lock(_fileLock)
{
    _dirFiles = new string[0];
    _fileIndex = 0;
}
if(_fileThread != null && _fileThread.IsAlive)
{
    _fileThread.Abort();
}
_fileThread = new Thread(() =>
    {
        string dir = Path.GetDirectoryName(fileName) ?? ".";
        lock (_fileLock)
        {
            _dirFiles = GetImageFileExtensions().SelectMany(f => Directory.GetFiles(dir, f, _searchOption)).OrderBy(f => f).ToArray();
            _fileIndex = Array.IndexOf(_dirFiles, fileName);
        }
        int totalFileCount = Directory.GetFiles(dir, "*.*", _searchOption).Length;

        Invoke((MethodInvoker)delegate
        {
            lblFileNbr.Text = string.Format("{0}/{1}", NumberFormat(_fileIndex + 1), NumberFormat(_dirFiles.Length));
            lblFileNbr.ToolTipText = string.Format("{0} ({1} files ignored)", dir, NumberFormat(totalFileCount - _dirFiles.Length));
        });
    });
_fileThread.Start();

我正在构建一个小的图像查看程序。打开图像时,它会列出同一目录中的文件数。我注意到,当我在包含许多其他文件(比如 150K)的目录中打开图像时,构建文件列表需要几秒钟。因此,我将此任务委托给另一个线程。

但是,如果您在完成搜索文件之前打开另一个图像,则旧计数不再相关,因此我将中止线程。

我正在锁定_dirFiles_fileIndex因为我想添加一些关键功能LeftRight在照片之间切换,所以我必须在其他地方访问那些(但在 UI 线程中)。

这安全吗?现在 C# 中处理线程的方法似乎有几十种,我只是想要一些简单的方法。


fileName是一个局部变量(这意味着它将被“复制”到匿名函数中,对吗?),并且_searchOption是只读的,所以我想这两个是可以安全访问的。

4

2 回答 2

3

> 中止这个文件搜索线程是否安全?

简短的回答是否定的!

中止线程几乎是不安全的,当您可能正在执行本机代码时,此建议更适用。

如果您不能足够快地合作退出(因为您的调用Directory.GetFiles需要时间),您最好的选择是放弃线程:让它干净地完成但忽略它的结果。

一如既往,我推荐阅读Joe Albahari 的免费电子书

于 2012-06-10T19:16:59.847 回答
1

使用 Thread.Abort() 中止线程是不安全的。但是您可以改为实现自己的中止,这可以让您以受控方式安全地关闭线程。

如果使用 EnumerateFiles 而不是 GetFiles,则可以在递增计数器时循环遍历每个文件以获取文件总数,同时检查标志以查看线程是否需要中止。

调用诸如此类的东西来代替您当前的 GetFiles().Length:

private bool AbortSearch = false;
private int NumberOfFiles(string dir, string searchPattern, SearchOption searchOption)
{
    var files = Directory.EnumerateFiles(dir, searchPattern, searchOption);
    int numberOfFiles = 0;
    foreach (var file in files)
    {
        numberOfFiles++;

        if (AbortSearch)
        {
            break;
        }
    }
    return numberOfFiles;
}

然后你可以替换

_fileThread.Abort();

AbortSearch=true;
_fileThread.Join();

您将使用当前的 Thread.Abort() 实现您的目标,但您将允许所有线程在您希望的时候干净地结束。

于 2012-06-11T10:55:40.487 回答