5

我在我的 Windows 8 应用程序中有一些数据,这些数据应该是附带的,只是一些静态数据。事实上:这是一个应该反序列化的简单 xml 文件。

数据保存在 Assets\data.xml 中(Assets 是空白应用模板中的默认文件夹)。

我正在使用这段代码来访问它:

private static async Task<MyObject> Load()
{
    if (Windows.ApplicationModel.DesignMode.DesignModeEnabled)
    {
        return new SampleData();
    }

    var uri = new Uri("ms-appx:///Assets/data.xml");
    Debug.WriteLine("Getting file from Application");
    var file = await StorageFile.GetFileFromApplicationUriAsync(uri);
    Debug.WriteLine("Opening file for reading async");
    var stream = await file.OpenStreamForReadAsync();

    var serializer = new XmlSerializer(typeof(MyObject));

    Debug.WriteLine("Begin deserialization");
    var result = (MyObject)serializer.Deserialize(stream.AsInputStream().AsStreamForRead());

    return result;
}

调用方法:

public static MyObject GetMyObject()
{
    if (_myObject == null)
    {
        _myObject = Load().Result;
    }

    return _myObject;
}

关于这一点的“有趣”部分是:

如果我在 处设置断点var uri = new Uri(...);并使用 F11 单步执行代码,一切都会按预期工作。我得到了所有的调试行,我的应用程序根据需要显示了静态数据。

如果我不设置断点并且不跳过这段代码,我只会得到 Debug 输出,Getting a file from Application而不会发生任何事情。似乎GetFileFromApplicationUriAsync()再也回不来了。我等了五分钟多,但仍然没有任何反应。

有人知道吗?

4

1 回答 1

12

感谢您发布代码。请尝试更改您的方法Load如下:

//your code
var file = await StorageFile.GetFileFromApplicationUriAsync(uri).AsTask().ConfigureAwait(false);
//your code
var stream = await file.OpenStreamForReadAsync().ConfigureAwait(false);
//your code

这里的主要区别是AsTask().ConfigureAwait(false)

编辑:

很高兴听到它正在工作。解释很简单:当您使用task.Resulttask.Wait()在 GUI 线程上与await关键字一起使用时,您会导致死锁。发生这种情况是因为在awaiting代码在其被调用的同一上下文中恢复之后(在您的情况下 - GUI 线程)。并且由于 GUI 线程当前正在等待任务完成(通过ResultWait())出现死锁,并且await永远不会调用关键字之后的代码。ConfigureAwait(false)指定可以忽略当前上下文,从而允许您的代码成功完成。更多细节在这里:http: //blog.stephencleary.com/2012/07/dont-block-on-async-code.html

于 2012-09-04T05:52:58.720 回答