我遇到了“引用”一词,我试图找出一些使用它的真实例子。每个代码表达式都有 AST 的能力听起来很棒,但是如何在现实生活中使用它呢?
有谁知道这样的例子?
我遇到了“引用”一词,我试图找出一些使用它的真实例子。每个代码表达式都有 AST 的能力听起来很棒,但是如何在现实生活中使用它呢?
有谁知道这样的例子?
F# 和 Nemerle 引用都用于元编程,但方法不同:Nemerle 在编译时使用元编程来扩展语言,而 F# 在运行时使用它们。
在 Nemerle 中,在宏中使用引号来分解代码片段并生成新代码。许多语言本身都是以这种方式实现的。例如,这里有一个来自官方库的示例——实现when
条件构造的宏。Nemerle 没有语句,因此 anif
必须有一个else
part:when
并且unless
宏分别为 anif
提供了一个空的then
和else
part 的简写。该when
宏还具有扩展的模式匹配功能。
macro whenmacro (cond, body)
syntax ("when", "(", cond, ")", body)
{
match (cond)
{
| <[ $subCond is $pattern ]> with guard = null
| <[ $subCond is $pattern when $guard ]> =>
match (pattern)
{
| PT.PExpr.Call when guard != null =>
// generate expression to replace 'when (expr is call when guard) body'
<[ match ($subCond) { | $pattern when $guard => $body : void | _ => () } ]>
| PT.PExpr.Call =>
// generate expression to replace 'when (expr is call) body'
<[ match ($subCond) { | $pattern => $body : void | _ => () } ]>
| _ =>
// generate expression to replace 'when (expr is pattern) body'
<[ match ($cond) { | true => $body : void | _ => () } ]>
}
| _ =>
// generate expression to replace 'when (cond) body'
<[ match ($cond : bool) { | true => $body : void | _ => () } ]>
}
}
该代码使用引号来处理看起来像一些预定义模板的模式并将它们替换为相应的match
表达式。例如,cond
将给定宏的表达式与:
<[ $subCond is $pattern when $guard ]>
检查它是否遵循x is y when z
模式并为我们提供组成它的表达式。如果匹配成功,我们可以从我们得到的部分生成一个新的表达式:
<[
match ($subCond)
{
| $pattern when $guard => $body : void
| _ => ()
}
]>
这将转换when (x is y when z) body
为基本的模式匹配表达式。所有这些都是自动类型安全的,并且在使用不正确时会产生合理的编译错误。因此,正如您所见,引用提供了一种非常方便且类型安全的代码操作方式。
我认为引文在 F# 和 Nemerle 中有完全不同的用途。在 F# 中,您不使用引号来扩展 F# 语言本身,而是使用它们来获取用标准 F# 编写的某些程序的 AST(代码的数据表示)。
在 F# 中,这可以通过将一段代码包装在 中来完成<@ ..F# code.. @>
,或者通过向函数添加特殊属性来完成:
[<ReflectedDefinition>]
let foo () =
// body of a function (standard F# code)
Robert 已经提到了这种机制的一些用途——您可以将代码转换为 F# 到SQL 以查询数据库,但还有其他几种用途。例如,您可以:
正如 Jordão 所提到的,引用已经启用元编程。一个现实世界的例子是使用引号将 F# 翻译成另一种语言的能力,例如 SQL。通过这种方式,报价服务与 C# 中的表达式树的用途大致相同:它们使 linq 查询能够被转换为 SQL(或其他数据访问语言)并针对数据存储执行。
Unquote是引用用法的真实示例。