3

我有一些方法可以使用 MemoryMappedFiles 来写入/读取数据。如果我使用简单的字符串作为文件名,它们就可以正常工作,例如“file.mmf”。但是,如果我使用完整目录路径,则会引发上述异常 - Exception has been thrown by the target of an invocation. 带有内部异常 - {"Could not find a part of the path."}。这是我的方法的样子:

public void WriteToFile(string fileName, string value)
{
    string newFileName = CombineDirectory(fileName);
    byte[] newValue = Encoding.UTF8.GetBytes(value);
    long capacity = newValue.Length + INT_MAXVALUE_TO_BYTEARRAY_LENGTH;

    using (var mmf = MemoryMappedFile.CreateFromFile(newFileName, FileMode.Create, newFileName, capacity))
    {
        using (var accesor = mmf.CreateViewAccessor())
        {
            byte[] newValueLength = BitConverter.GetBytes(value.Length);
            accesor.WriteArray(0, newValueLength, 0, newValueLength.Length);
            accesor.WriteArray(INT_MAXVALUE_TO_BYTEARRAY_LENGTH, newValue, 0, newValue.Length);
        }
    }
}

我的路径如下所示:

"C:\\Users\\MyUser\\Documents\\Visual Studio 2012.mmf"

我正在使用

Path.Combine

异常发生在第一个“使用”行。如果我尝试使用相同的文件路径创建一个文件

File.Create

该文件正在创建没有问题。

如果有人有任何建议,那就太好了。

问候

4

1 回答 1

5

您需要确保mapName参数(即调用中的第三个参数CreateFromFile)与文件路径不同。PathNotFound如果你这样做,它会抛出一个异常。我同意,这对找出失败的原因并没有真正的帮助。

因此,您选择地图名称值的选项:

  • 生成一些唯一的密钥,例如Guid.NewGuid().ToString()
  • 使用一个常数值,例如“MySpecialMapForThings”
  • 使用一些约定,例如生成一个唯一键,您也可以将它用于映射文件的文件名部分。

最后一个选项的示例:

public static Tuple<FileInfo, string> GenerateMapInfo(string mapDirectory, string fileExtension)
{
    var uniqueMapName = Guid.NewGuid().ToString();
    var fileName = Path.Combine(mapDirectory, Path.ChangeExtension(uniqueMapName, fileExtension));
    return Tuple.Create(new FileInfo(fileName), uniqueMapName);
}

public void WriteToFile(Tuple<FileInfo, string> mapInfo, string value)
{
    byte[] newValue = Encoding.UTF8.GetBytes(value);
    long capacity = newValue.Length + INT_MAXVALUE_TO_BYTEARRAY_LENGTH;

    using (var mmf = MemoryMappedFile.CreateFromFile(mapInfo.Item1.FullName, FileMode.Create, mapInfo.Item2, capacity))
    using (var accesor = mmf.CreateViewAccessor())
    {
        byte[] newValueLength = BitConverter.GetBytes(value.Length);
        accesor.WriteArray(0, newValueLength, 0, newValueLength.Length);
        accesor.WriteArray(INT_MAXVALUE_TO_BYTEARRAY_LENGTH, newValue, 0, newValue.Length);
    }
}
于 2013-08-31T21:11:01.023 回答