如何解决这个僵局?

时间:2017-05-14 13:47:18

标签: c# uwp win-universal-app deadlock

我有一个UWP应用程序,我试图在本地文本文件中存储和检索一些数据,但无论我如何尝试这样做,应用程序都会死锁。由于需要发生同步的事情,我尝试使用任务并等待其完成但是,应用程序锁定。

我的UWP应用程序中有一个名为“MainPage”的页面,在它的构造函数中我有以下代码:

var listenkeyViewModel = new ListenkeyViewModel();
listenkeyViewModel.GetKey();
listenkey = listenkeyViewModel.Listenkey;

get键是这里的问题,因为它调用ViewModel上的一个方法(我创建它是同步的,因为我认为调用同步asap会更好。

public void GetKey()
{
    try
    {
        var listenKeyTask = RetrieveListenKey();
        _listenkey = listenKeyTask.Result;
    }
    catch (Exception e)
    {
    }
}

public static async Task<string> RetrieveListenKey()
{
    try
    {
        var storageFolder = Windows.Storage.ApplicationData.Current.LocalFolder;
        var listenkeyFile = await storageFolder.GetFileAsync("listenkey.txt");
        return await Windows.Storage.FileIO.ReadTextAsync(listenkeyFile);
    }
    catch (Exception e)
    {
        throw new Exception("Could not load file");
    }
}

我知道事情是“一直异步”,但这不可能。我无法使原始代码异步的构造函数。你怎么会不陷入僵局?我不明白。

1 个答案:

答案 0 :(得分:3)

GetKey转换为async / await

public async Task GetKey() {
    try {
        var listenKeyTask = RetrieveListenKey();
        _listenkey = await listenKeyTask;
    } catch (Exception e) {
        //...should handle/log error
    }
}

将此调用移出构造函数并移入事件处理程序。比如页面加载或在页面生命周期的早期调用的其他事件。

partial class MainPage : Page {
    ListenkeyViewModel listenkeyViewModel;
    string listenkey;

    public MainPage() {
        InitializeComponent();
        listenkeyViewModel = new ListenkeyViewModel();
        // add a handler to be called when the page has been loaded
        this.Loaded += OnPageLoaded;             
    }

    async void OnPageLoaded(object sender, RoutedEventArgs e) {            
        await listenkeyViewModel.GetKey();
        listenkey = listenkeyViewModel.Listenkey;
    }

    // Shown for demonstration purposes only.
    // This is typically autogenerated by Visual Studio.
    private void InitializeComponent() {
    }
}
事件处理程序允许

async void,因此这应该允许进程在没有死锁的情况下流动。