UWP在发布模式下丢失序列化数据

时间:2017-03-12 16:29:03

标签: debugging uwp release

我正在尝试使用Visual Studio Community 2015在UWP中创建日记应用程序。所以所有日记条目都保存在ObservableCollection中(类型为DiaryEntry,它有三个字符串)。在App类的OnSuspending方法中,我序列化(使用XmlSerializer)并将ObservableCollection保存到应用程序的LocalFolder。在App类的OnLaunched方法中,我反序列化数据并将其再次放在ObservableCollection中。

当我在调试模式下运行此应用程序时,它每次都是100%工作,但在发布模式下,一旦我关闭并再次打开应用程序,我就会丢失ObservableCollection。其他时间它工作一次或两次但我仍然丢失数据。我想知道的是如何让我的应用程序在发布模式下工作?

这些是我正在使用的方法:

        private async void SaveCollection(string xml)
    {
        //Serializing our observablecollection and saving it to the local folder
        StorageFile sf = await ApplicationData.Current.LocalFolder.CreateFileAsync("Diary.txt", CreationCollisionOption.ReplaceExisting);
        await Windows.Storage.FileIO.WriteTextAsync(sf, xml);

    }

    private async Task<string> GetSavedCollection()
    {
        try
        {
            Windows.Storage.StorageFolder storageFolder = Windows.Storage.ApplicationData.Current.LocalFolder;
            Windows.Storage.StorageFile sampleFile = await storageFolder.GetFileAsync("Diary.txt");

            string text = await Windows.Storage.FileIO.ReadTextAsync(sampleFile);

            return text;
        }
        catch (FileNotFoundException e)
        {
            return "";
        }

    }

    public static string ToXml(ObservableCollection<DiaryEntry> d)
    {
        XmlSerializer serializer = new XmlSerializer(typeof(ObservableCollection<DiaryEntry>));
        StringBuilder stringBuilder = new StringBuilder();
        XmlWriterSettings settings = new XmlWriterSettings()
        {
            Indent = true,
            OmitXmlDeclaration = true,
        };

        using (XmlWriter xmlWriter = XmlWriter.Create(stringBuilder, settings))
        {
            serializer.Serialize(xmlWriter, d);
        }
        return stringBuilder.ToString();
    }

    // Deserialize from xml 
    public static ObservableCollection<DiaryEntry> FromXml(string xml)
    {
        if (xml == "")
        {
            return new ObservableCollection<DiaryEntry>();
        }

        XmlSerializer serializer = new XmlSerializer(typeof(ObservableCollection<DiaryEntry>));
        ObservableCollection<DiaryEntry> value;
        using (StringReader stringReader = new StringReader(xml))
        {
            object deserialized = serializer.Deserialize(stringReader);
            value = (ObservableCollection<DiaryEntry>)deserialized;
        }

        return value;
    }

这就是我称之为这些方法的方法:

Diaries = FromXml(等待GetSavedCollection());

SaveCollection(toxml用于(日记));

1 个答案:

答案 0 :(得分:0)

如果没有对代码进行全面复制,很难说是肯定的(请参阅https://stackoverflow.com/help/mcve),但我怀疑它是否会误解应用生命周期以及您添加的位置您的代码是OnLaunchedOnSuspending方法。

要了解生命周期,请参阅the docs herethis blog article

现在这些方法。

让我们首先处理OnSuspending,因为它更简单。您的代码应如下所示:

private void OnSuspending(object sender, SuspendingEventArgs e)
{
    var deferral = e.SuspendingOperation.GetDeferral();

    SaveCollection(ToXml(Diaries));

    deferral.Complete();
}

请注意,如果您在SaveCollection之后调用deferral.Complete();,则可能会在应用终止之前完成,并且可能是没有

的原因

OnLaunching方法更复杂。我怀疑你有这样的代码:

if (e.PreviousExecutionState == ApplicationExecutionState.Terminated)
{
    //TODO: Load state from previously suspended application
    Diaries = FromXml(await GetSavedCollection());
}

根据模板中的TODO注释,这似乎是合乎逻辑的做法,但对于您希望在重新启动应用时始终从磁盘加载的场景不合适。例如,上述不会在应用程序先前由用户关闭时加载内容。我假设你也想要加载日记条目。 (因为您还没有提供完整的重复步骤,我无法确切地知道您如何测试并重新创建您所看到的行为。)

相反,如果需要,请在'PrelaunchActivated检查之前加载日记。

if (Diaries == null)
{
    Diaries = FromXml(await GetSavedCollection());
}

if (e.PrelaunchActivated == false)

这是您定义和调用SaveCollection的方式的问题。

您已将SaveCollection定义为async void。这是不好的。唯一应该是async void的事件是事件处理程序。您应该将其定义为async Task并等待对它的调用:

private async Task SaveCollection(string xml)
{
    System.Diagnostics.Debug.WriteLine(xml);

    //Serializing our observablecollection and saving it to the local folder
    StorageFile sf = await ApplicationData.Current.LocalFolder.CreateFileAsync("Diary.txt", CreationCollisionOption.ReplaceExisting);
    await Windows.Storage.FileIO.WriteTextAsync(sf, xml);
}


await SaveCollection(ToXml(Diaries));

无需等待此调用,可能无法正确完成。在.NetNative编译下这种行为更加明显不会让我感到惊讶。

或者它可能是两个问题的组合。