如何在UWP中覆盖文件?

9
我尝试使用FileSavePicker在UWP中覆盖当前文件。我参考了这个链接:https://msdn.microsoft.com/zh-cn/windows/uwp/files/quickstart-reading-and-writing-files。但是当我使用一个较短的文件来覆盖一个较长的文件时,会出现一个问题:旧文件(较长)的额外部分没有被删除,就像这样:

旧文件:

abcdefg

我想保存的文件:

hij

但是在保存操作之后,我得到了以下结果:
hijdefg

这是我的代码:

        private async void Save_Click(object sender, RoutedEventArgs e)
    {
        XmlDocument Line = SaveToXml();

        //Save it
        FileSavePicker fileSaving = new FileSavePicker();
        fileSaving.SuggestedStartLocation = PickerLocationId.DocumentsLibrary;
        fileSaving.SuggestedFileName = "NewLine";
        fileSaving.FileTypeChoices.Add("Simple Line Files", new List<string>() { ".Line" });
        StorageFile file = await fileSaving.PickSaveFileAsync();

        if (file != null)
        {
            CachedFileManager.DeferUpdates(file);
            var stream = await file.OpenAsync(Windows.Storage.FileAccessMode.ReadWrite);
            //var stream = await file.OpenStreamForWriteAsync();
            using (var outputStream = stream.GetOutputStreamAt(0))
            {
                using (var dataWriter = new Windows.Storage.Streams.DataWriter(outputStream))
                {
                    string xmlDoc = Line.OuterXml;
                    Debug.WriteLine(Line.OuterXml);
                    dataWriter.WriteString(xmlDoc);
                    await dataWriter.StoreAsync();
                    await outputStream.FlushAsync();
                }
            }
            stream.Dispose();
        }
    }

谢谢你帮助我!


1
似乎是正常行为。如果您想用全新的数据覆盖文件,为什么不使用ReplaceExisting选项创建文件,而不是打开旧文件呢? - Romasz
2
错误的方法,您想要创建而不是打开。 - Hans Passant
4个回答

10

在开始写入之前,您还可以立即调用 Stream.SetLength(0) 来清除流。

var streamToWrite = await file.OpenStreamForWriteAsync().ConfigureAwait(false);
streamToWrite.SetLength(0);
using (var streamWriter = new StreamWriter(streamToWrite))
{
    streamWriter.Write(data);
    streamWriter.Flush();                        
}

1
你的代码 var outputStream = stream.GetOutputStreamAt(0) 返回文件流中开始位置的输出流。 当你写入文件时,它将从你设置的位置替换文件。如果要写入的新字符串比文件中的旧字符串短,则无法替换所有旧字符串。因此,使用此方法,无法确保每个字母都可以在旧文件中被替换。
你可以使用 FileIO.WriteTextAsync(IStorageFile, String) 方法将文本写入指定的文件。它可以覆盖旧文件。
例如:
private async void Save_Click(object sender, RoutedEventArgs e)
{
    FileSavePicker savePicker = new FileSavePicker();
    savePicker.SuggestedStartLocation = PickerLocationId.DocumentsLibrary;
    savePicker.FileTypeChoices.Add("Simple Line Files", new List<string>() { ".line" });
    savePicker.SuggestedFileName = "NewLine";
    StorageFile file = await savePicker.PickSaveFileAsync();
    if (file != null)
    {
        CachedFileManager.DeferUpdates(file);
        await FileIO.WriteTextAsync(file, "hello");
        FileUpdateStatus status = await CachedFileManager.CompleteUpdatesAsync(file);
    }
}

1
如果我不是在写文本,而是使用DataWriter,该怎么办?请问如何解决这个问题? - M.kazem Akhgary

0
在调用OpenStreamForWriteAsync()之前,使用FileIO.WriteBytesAsync并写入一个零字节数组。这将清除文件。
public async Task<Stream> ClearAndGetStream(StorageFile storageFile)
{
    await FileIO.WriteBytesAsync(storageFile, new byte[0]);
    return await storageFile.OpenStreamForWriteAsync();
}

0

在仍然使用DataWriter的情况下,有一个解决方案。我和你一样遇到了完全相同的问题,并通过以下方法解决了它。

替换为:

await dataWriter.StoreAsync();

使用:

stream.size = await dataWriter.StoreAsync();

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接