在 C# 中我曾经写过

var provider = A.Fake<ITimeProvider>();
A.CallTo(() => provider.Fetch()).ReturnsLazily(call => data[0]);
container.Register(() => provider);

捕获对 Fetch() 的调用。

当我用 F# 尝试过同样的事情时
let provider = A.Fake<ITimeProvider>()
A.CallTo(fun () -> provider.Fetch()).ReturnsLazily(fun call -> data.[0]) |> ignore
container.Register(fun () -> provider)

测试失败
Test Error : ....Test
System.ArgumentException : The specified object is not recognized as a fake object.
at Microsoft.FSharp.Control.AsyncBuilderImpl.commit[a](Result`1 res)
at Microsoft.FSharp.Control.CancellationTokenOps.RunSynchronously[a](CancellationToken token, FSharpAsync`1 computation, FSharpOption`1 timeout)
at Microsoft.FSharp.Control.FSharpAsync.RunSynchronously[T](FSharpAsync`1 computation, FSharpOption`1 timeout, FSharpOption`1 cancellationToken)
...

对于 F#,它的定义应该有点不同。你知道怎么做吗?

最佳答案

FakeItEasy 使用 F# 3 支持的 LINQ 表达式,但是在使用静态 API 时似乎不兼容。根据错误消息“对象未被识别为假对象”,我怀疑在这种情况下对象的分辨率是 C#/VB.Net 特定的。

然而,FakeItEasy 基于实例成员的设置似乎确实有效:

let fake = Fake<ITimeProvider>()
fake.CallsTo(fun x -> x.Fetch()).ReturnsLazily(fun () -> data.[0]) |> ignore
let provider = fake.FakedObject

另一种选择是使用 F# 和 Moq(这里我使用的是 Moq.FSharp.Extensions ):
let mock = Mock<ITimeProvider>()
mock.SetupFunc(fun x -> x.Fetch()).Returns(data.[0]).End
let provider = mock.Object

或者 Foq ,它是专门为 F# 设计的:
let provider =
    Mock<ITimerProvider>.Method(fun x -> <@ x.Fetch @>).Returns(data.[0])

关于c# - A.CallTo(...).ReturnsLazily(...) 抛出 "The specified object is not recognized as a fake object.",我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/32052187/

10-15 15:50