- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
在 F# 中编写测试时,我试图生成有关导致错误的状态的有用消息。在 python 中,我会包含所有的 locals(),因此它们可以在测试跟踪中轻松访问。
F# 中是否有类似的结构?
我一直在网上搜索优秀的 fsharpforfunandprofit 网站,并浏览了保留关键字列表。
这是我希望能够执行的一些代码。
[<Property>]
let some_test x =
let example_inner_helper y =
let z = y + x
// Example of the meta-construction i am looking for
let scope = {|
local = {| y = y; z = z |};
capture = {|
local = {| x = x |};
capture = {| (* ... *) |};
|};
|}
x = z |@ sprintf "require %A=%A (%A)" x y scope
example_inner_helper (x+x)
这会产生非常有用的输出
Tests.some_test
Source: Tests.fs line 105
Duration: 51 ms
Message:
FsCheck.Xunit.PropertyFailedException :
Falsifiable, after 1 test (1 shrink) (StdGen (387696160,296644521)):
Label of failing property: require 1=2 ({ capture = { capture = {}
local = { x = 1 } }
local = { y = 2
z = 3 } })
Original:
-1
Shrunk:
1
但是,我必须明确地捕获像“作用域”这样的构造可以自动提供的信息。导致丑陋、容易出错的东西,比如
let ``firstAscendingLastDescendingPartE Prop`` (a: Extent<int>) b =
let validateT ea eb =
let checkXbeforeY ex ey headx restx =
match restx with
| ValueNone -> // No rest
(ex = headx) |@ sprintf "no rest: ex = headx (%A = %A)" ex headx
.&. ((intersectE ex ey) = ValueNone) |@ sprintf "no rest: ex ∩ ey = ∅ (%A ∩ %A)" ex ey
| ValueSome rex -> // Some rest ->
// headx and restx combines to ex
((expandE headx rex) = ex) |@ sprintf "rest: headx+rex = ex (%A...%A)" headx rex
.&. (exactlyTouchesE headx rex) |@ sprintf "rest: headx ends where rex starts (ex=%A ey=%A headx=%A restx=%A)" ex ey headx restx
.&. (exactlyTouchesE headx ey) |@ sprintf "rest: headx ends where ey starts (ex=%A ey=%A headx=%A restx=%A)" ex ey headx restx
....
(当然,我不关心“作用域”构造的实际类型)
我已经看过 unquote 了,它很可爱,看起来也不错。但它在很大程度上限制了代码:
我的代码看起来有点像:
[<Struct>]
type X<'T when 'T: comparison and 'T: equality> =
val public first: 'T
val public last: 'T
new (first: 'T, last: 'T) = {
first =
(if last <= first then
invalidOp (sprintf "first < last required, (%A) is not < (%A)" first last)
else first)
last = last;
}
// ..
所以我在使用以下两种结构进行测试时遇到问题(导致上述错误)。
let quote_limits () =
let x first last = X(first, last)
let validate (x: X<'a>) = x.first < x.last
let a = X(0, 1)
<@ a.first = 0 @> |> test
<@ validate a @> |> test
第一个我可以解决访问结构部分的函数,但对泛型的限制是 PITA。
最佳答案
我认为没有办法做到这一点,即使在原则上也是如此。 F# 代码被编译为基于堆栈的 .NET 字节码,因此局部变量实际上(通常)不存在于编译代码中。您可能能够获得一些全局变量(它们是静态成员)或者可能以某种方式使用调试信息,但我认为没有标准的方法可以做到这一点。
就是说,在您的特定情况下,F# 实际上有一个不错的选择,即使用 unquote测试框架。这使您可以将测试代码放在 F# 引用中,然后它会显示该引用的计算方式。例如:
let some_test x =
let example_inner_helper y =
<@
let z = y + x
x = z
@> |> test
example_inner_helper (x+x)
some_test 10
在这里,我在引用中定义了局部变量z
。当您运行测试时,unquote 将打印各个评估步骤,您可以在那里看到 z
的值:
Test failed:
let z = y + x in x = z
let z = 20 + 10 in x = z
let z = 30 in x = z
false
关于testing - F# 是否具有访问词法作用域的语言构造(如 python locals()/globals()),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/57913947/
我试图了解传递给 setTimeout 的箭头函数如何记住上一个执行上下文中的 this 的值。我知道在执行箭头函数时会使用词法作用域规则查找 this 值。这是否意味着箭头函数关闭变量和 this
这个问题已经有答案了: How does the "this" keyword in Javascript act within an object literal? [duplicate] (4 个
我已阅读 this问题,我想我已经理解了投票最多的答案,但他说 since basically every programming language in wide use today uses le
如何让这段宏发挥预期的作用? -- 我想从词法环境中捕获 p 而不必将其作为参数发送给宏。 (define-syntax-rule (fi a b) (if p a b)) ;--->capt
Program A() { x, y, z: integer; procedure B() { y: integer; y=0;
我正在用 Java 实现自己的链表。节点类只有一个名为“name”的字符串字段和一个名为“link”的节点。现在我有一个测试驱动程序类,它只按顺序插入几个名字。现在,我正在尝试编写一种排序方法来按字母
考虑到这个question SO,其中调用了整个 C# 内存中编译器。只有lexical and syntactic analyzing时是必需的:将文本解析为词素流,检查它们并退出。 在System
我有 2 个场景。 这失败了: class F { public X X { get; set; } } 错误 CS0102:类型“F” ' 已经包含 ' X 的定义| ' 这个有效: class
我有一个用 NodeJS 执行的 .js 文件。这是我的文件的内容: var ctry = "America"; function outer(msg) { console.log(msg +
我对编写汇编程序的概念非常陌生,即使在阅读了大量 Material 之后,我仍然很难理解几个概念。 将源文件实际分解为 token 的过程是什么?我相信这个过程称为词法分析,我已经到处搜索有意义的真实
在 static scoping,标识符可以通过分析/解析源代码来确定(与动态作用域不同,动态作用域或多或少需要了解调用者环境)。 我的问题是这样的,因为静态作用域只需要解析源代码以了解作用域和标识符
编辑:我在第一个答案后更改了示例代码,因为我想出了一个简单的版本来回避相同的问题。 我目前正在学习 Common Lisp 的作用域属性。在我认为我有一个坚实的理解之后,我决定编写一些我可以预测结果的
考虑这段代码: class Bar(object): pass class Foo(object): def bar(self): return Bar() f = Foo() def Bar
将 ES6 箭头函数与词法 this 绑定(bind)结合使用非常棒。 但是,我刚才在使用典型的 jQuery 单击绑定(bind)时遇到了一个问题: class Game { foo() {
将 ES6 箭头函数与词法 this 绑定(bind)结合使用非常好。 但是,我刚才在将它与典型的 jQuery 点击绑定(bind)一起使用时遇到了一个问题: class Game { foo(
我是一名优秀的程序员,十分优秀!