如何检查IOException是否足够 - 磁盘空间异常类型?

时间:2012-02-15 12:18:40

标签: c# .net exception-handling ioexception diskspace

如何检查IOException是否为“磁盘空间不足”异常类型?

目前我检查消息是否与“磁盘空间不足”相匹配,但我知道如果操作系统语言不是英语,这将不起作用。

5 个答案:

答案 0 :(得分:72)

您需要查看HResult并针对ERROR_DISK_FULL (0x70)ERROR_HANDLE_DISK_FULL (0x27)进行测试,can be converted to HResults by OR'ing with 0x80070000

对于.Net Framework 4.5及更高版本,您可以使用Exception.HResult属性:

static bool IsDiskFull(Exception ex)
{
    const int HR_ERROR_HANDLE_DISK_FULL = unchecked((int)0x80070027);
    const int HR_ERROR_DISK_FULL = unchecked((int)0x80070070);

    return ex.HResult == HR_ERROR_HANDLE_DISK_FULL 
        || ex.HResult == HR_ERROR_DISK_FULL;
}

对于旧版本,您可以使用Marshal.GetHRForException取回HResult,但has significant side-effects and is not recommended

static bool IsDiskFull(Exception ex)
{
    const int ERROR_HANDLE_DISK_FULL = 0x27;
    const int ERROR_DISK_FULL = 0x70;

    int win32ErrorCode = Marshal.GetHRForException(ex) & 0xFFFF;
    return win32ErrorCode == ERROR_HANDLE_DISK_FULL || win32ErrorCode == ERROR_DISK_FULL;
}

来自MSDN文档:

  

请注意, GetHRForException 方法设置 IErrorInfo   当前线程。这可能会导致类似的方法出现意外结果   默认使用 IErrorInfo ThrowExceptionForHR方法   当前线程如果已设置。

另见How do I determine the HResult for a System.IO.IOException?

答案 1 :(得分:20)

在.NET 4.5中,HResult属性getter现在是Public,因此您不必再使用Marshal.GetHRForException(及其副作用)。

http://msdn.microsoft.com/en-us/library/system.exception.hresult(v=vs.110).aspx状态"从.NET Framework 4.5开始,HResult属性的setter受到保护,而其getter是公共的。在以前版本的.NET Framework中,getter和setter都受到保护"

因此,您可以使用Justin的答案,但将Marshal.GetHRForException(ex)替换为ex.HResult

答案 2 :(得分:13)

嗯,这有点笨拙,但我们走了。

首先要做的是从异常中获取HResult。由于它是受保护的成员,我们需要一些反思才能获得价值。这是一个扩展方法,可以解决这个问题:

public static class ExceptionExtensions
{
    public static int HResultPublic(this Exception exception)
    {
        var hResult = exception.GetType().GetProperties(BindingFlags.NonPublic | BindingFlags.Public | BindingFlags.Instance).Where(z => z.Name.Equals("HResult")).First();
        return (int)hResult.GetValue(exception, null);
    }
}

现在,在您的捕获范围内,您可以获得HResult

catch (Exception ex)
{
    int hResult = ex.HResultPublic();
}

从这里开始,你必须解释HResult。您需要this link

我们需要得到存储在值的16位的ErrorCode,所以这里有一些位操作:

int errorCode = (int)(hResult & 0x0000FFFF);

现在,请参阅the list of system error codes,我们在这里:

ERROR_DISK_FULL
112 (0x70)

所以使用以下方法测试:

switch (errorCode)
{
    case 112:
        // Disk full
}

也许有一些“更高级别”的功能来获得所有这些东西,但至少它可以工作。

答案 3 :(得分:10)

最简单的内联解决方案(min .NET 4.5和C#6):

try
{
    //...
}
catch (IOException ex) when ((ex.HResult & 0xFFFF) == 0x27 || (ex.HResult & 0xFFFF) == 0x70)
{
    //...
}

答案 4 :(得分:1)

System.IOException有许多派生的Exception类型,但是这些派生类型中没有一个听起来像你的例外。您可以查看异常的HResult或Data属性,也许这将具有更详细的更具体的错误代码。根据{{​​3}},这些属性都是该异常类型的一部分。只需确保您尝试捕获特定的异常类型,而不仅仅是基本的异常类型。