4

我在 F# 中的视图模型

我正在尝试使用 F# 而不是 C# 来实现我的 ViewModel。我正在关注这篇文章(顺便说一句,有没有更新或更好的建议?)。

所以假设我有我的视图模型基础实现(MVVM.ViewModel,它在 C# 中,但我可以从 F# 引用它)和一个简单的Status属性。

namespace FuncViewModel
open MVVM.ViewModel
open System

    type MyFuncViewModel() = 
        inherit ViewModelBase()

        let mutable status=""

        member this.RunSetStatus() =
            status <- "Reset @" + DateTime.Now.ToString "yy.MM.dd hh:mm:ss"
            base.OnPropertyChanged("Status")

        member this.SetStatus = new DelegateCommand(fun _ -> this.RunSetStatus() )


    member this.Status 
        with get() =
            status
        and set(value) =
             status <- value
             base.OnPropertyChanged(fun () -> this.Status)

一切都按预期工作,到目前为止一切都很好(但是如果您发现任何概念错误或者您是否找到上述代码的更惯用的版本,请告诉我)

介绍异步/等待模式

这就是我出错的地方:我知道如何在 C# 中做到这一点,但我在 F# 中并不擅长。

我已经尝试过以下方法。

member this.RunSetStatus() =
    status <- "Start resetting @" + DateTime.Now.ToString "yy.MM.dd hh:mm:ss"
    base.OnPropertyChanged("Status")
    let task = async {

        do! Async.Sleep (30 * 1000) 

    }
    Async.StartImmediate(task)
    status <- "Reset done @" + DateTime.Now.ToString "yy.MM.dd hh:mm:ss"
    base.OnPropertyChanged("Status")

问题是 - 当我运行完整的 WPF 应用程序时 - 我看不到预期的延迟:最终状态直接进入输出。

如果我将上面的内容更改Async.StartImmediate(task)Async.RunSynchronously(task),我当然会看到正在进行的延迟,但是应用程序被冻结了,所以这不是我想要的。

如果我将其重新排列为

member this.RunSetStatus() =
    status <- "Start resetting @" + DateTime.Now.ToString "yy.MM.dd hh:mm:ss"
    base.OnPropertyChanged("Status")
    let task = async {

        do! Async.Sleep (30 * 1000) 

        status <- "Reset done @" + DateTime.Now.ToString "yy.MM.dd hh:mm:ss"
        base.OnPropertyChanged("Status")

    }
    Async.StartImmediate(task)

我收到一个错误

成员或对象构造函数“OnPropertyChanged”不可访问。私有成员只能从声明类型中访问。受保护的成员只能从扩展类型访问,不能从内部 lambda 表达式访问。

编辑(续)

最后,我也试过这个

member this.RunSetStatus() =
    status <- "Start resetting @" + DateTime.Now.ToString "yy.MM.dd hh:mm:ss"
    base.OnPropertyChanged("Status")
    let task = async {

        do! Async.Sleep (30 * 1000) 

    }
    Async.StartWithContinuations(task, 
        (fun _ -> this.Status <- "Reset done @" + DateTime.Now.ToString "yy.MM.dd hh:mm:ss"),
        (fun _ -> this.Status <- "Operation failed."),
        (fun _ -> this.Status <- "Operation canceled."))

但应用程序因 ArgumentException 而崩溃

堆栈跟踪

应用程序:MyFuncWPF.exe 框架版本:v4.0.30319 描述:进程因未处理的异常而终止。异常信息:System.ArgumentException 堆栈:在 MVVM.ViewModel.ViewModelBase.OnPropertyChanged[System.__Canon, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089] 在 FuncViewModel.MyFuncViewModel.set_Status(System.String) 在 . $MyVMLib+RunSetStatus@20.Invoke(Microsoft.FSharp.Core.Unit) 在 Microsoft.FSharp.Control.CancellationTokenOps+StartWithContinuations@1274[[System.__Canon, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089 ]].Invoke(System.__Canon) at .$Control.loop@430-52(Microsoft.FSharp.Control.Trampoline, Microsoft.FSharp.Core.FSharpFunc2<Microsoft.FSharp.Core.Unit,Microsoft.FSharp.Control.FakeUnitValue>) at Microsoft.FSharp.Control.Trampoline.ExecuteAction(Microsoft.FSharp.Core.FSharpFunc2) 在 System.Windows.Threading.ExceptionWrapper.InternalRealCall(System.Delegate, System.Object) 的 .$Control+-ctor@507.Invoke(System.Object)

编辑 2 - 发现问题

我不得不使用以下更简单的重载(根据此源代码OnPropertyChanged,它们都在 C# 中实现和工作)

    member this.Status 
        with get() =
            status
        and set(value) =
             status <- value
             base.OnPropertyChanged("Status")
4

1 回答 1

0

异常的原因是 F# 的函数不能表示为MemberExpression

    protected virtual void OnPropertyChanged<T>(Expression<Func<T>> selectorExpression)
    {
        if (selectorExpression == null)
            throw new ArgumentNullException("selectorExpression");
        MemberExpression body = selectorExpression.Body as MemberExpression;
        if (body == null)
            throw new ArgumentException("The body must be a member expression");
        OnPropertyChanged(body.Member.Name);
    }

在调试器中,您会看到您得到的异常实际上是 - “主体必须是成员表达式”。

您的第一个代码:

member this.RunSetStatus() =
    status <- "Reset @" + DateTime.Now.ToString "yy.MM.dd hh:mm:ss"
    base.OnPropertyChanged("Status")

因为您没有使用属性的设置器,所以可以工作Status

所以你需要使用不同的重载。

于 2016-10-27T19:58:57.723 回答