- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
给定一个 AST 的 DU(一个非常简单的表达式树)
type Expression =
| Add of list<Expression>
| Var of string
| Val of float
我想写一个运算符
+
这样我就可以写
let x = "s" + 2.0
并且有
x = Add [(Var "s"); (Val 2.0)]
此外,我希望它可以从其他程序集中使用(通过打开一些东西)。
let (+) a b =
Add [a; b]
x = (Var "s") + (Val 2.0)
但在这里我还是要包装
"s"
和
"2.0"
用手。我想避免这种包装。
type IToExpression =
abstract member ToExpression : Expression
type Expression with
member this.ToExpression = this
type System.String with
member this.ToExpression = Var this
type System.Double with
member this.ToExpression = Val this
let (+++)
(a : 'S when 'S :> IToExpression)
(b : 'T when 'T :> IToExpression) =
Add [a.ToExpression; b.ToExpression]
然后与静态解析的类型参数相同。
let (++) a b =
let a = (^t : (member ToExpression: Expression) a)
let b = (^t : (member ToExpression: Expression) b)
Add [a; b]
编辑:但正如本文中所指出的
answer (并且由于我的草率复制)这种方法需要更多的工作才能解决真正的问题。
++
和
+++
未能在想要的表达式中进行类型检查,即行
let x = "s" ++ 2.0
let x = "s" +++ 2.0
我通读了
最佳答案
这里有几个微妙的问题。
一、 SRTP 仅适用于 inline
职能。这是因为它们不能编译成 IL(因为 IL 不支持这种约束),所以必须在编译时解决它们。静态的。这就是为什么它们是“静态解决的”。 inline
关键字使编译器能够做到这一点。
let inline (+) a b = ...
二、这是什么泛型类型
^t
你在引用?它是什么类型的?我认为你的意思是它是
a
的类型和
b
,但你还没有这样声明它,所以它只是一些随机泛型类型。它需要绑定(bind)到参数:
let inline (+) (a: ^t) (b: ^t) = ...
三、从您的示例来看,您实际上的意思是
a
和
b
是不同的类型,不是吗?
let inline (+) (a: ^a) (b: ^b) = ...
四、扩展方法不计入静态解析类型,因此您不能定义
ToExpression
在
String
和
float
并期望它起作用。通常的技巧是在一个特殊的类上声明所有方法,该类仅用于保存这些方法:
type ToExpressionStub() =
static member ToExpression s = Var s
static member ToExpression f = Val f
然后人们可能会期望这会起作用:
let inline (+) (a: ^a) (b: ^b) =
let a = (ToExpression : (static member ToExpression: ^a -> Expression) a)
let b = (ToExpression : (static member ToExpression: ^b -> Expression) b)
Add [a; b]
但不,它不会。这是因为静态解析的约束不能应用于具体类型,只能应用于类型变量,如
^a
。或
^b
.那么该怎么办?
let inline (+) (dummy: ^c) (a: ^a) (b: ^b) =
let a = (^c : (static member ToExpression: ^a -> Expression) a)
let b = (^c : (static member ToExpression: ^b -> Expression) b)
Add [a; b]
然后我们必须传递一个值
ToExpressionStub()
每次通话:
let x = (+) (ToExpressionStub()) "foo" 2.0
当然,这非常不方便,因此我们将添加另一个带有三个参数的中间函数,并从运算符
(+)
调用它。 :
let inline doAdd (dummy: ^c) (a: ^a) (b: ^b) =
let a = (^c : (static member ToExpression: ^a -> Expression) a)
let b = (^c : (static member ToExpression: ^b -> Expression) b)
Add [a; b]
let inline (+) (a: ^a) (b: ^b) = doAdd (ToExpressionStub()) a b
差不多好了!但这也不太奏效:在线
let b = ...
我们得到一个警告“这个结构使代码不那么通用......”,然后在使用站点我们得到一个错误,我们不能使用
string
或
float
,取决于哪个先出现。
^a = ^b
.为了打破这种僵局,我们可以简单地改变约束的形状,使它们变得不同:
let inline doAdd (dummy: ^c) (a: ^a) (b: ^b) =
let a = ((^a or ^c) : (static member ToExpression: ^a -> Expression) a)
let b = ((^b or ^c) : (static member ToExpression: ^b -> Expression) b)
Add [a; b]
请注意,它们现在应用于
^a or ^c
和
^b or ^c
分别,不仅仅是
^c
独自的。这还有一个额外的效果:我们不再局限于我们在
ToExpressionStub
中列举的类型。 .我们可以使用任何有自己的非扩展方法的类型
ToExpression
定义。例如,
Expression
本身:
type Expression =
| Add of list<Expression>
| Var of string
| Val of float
with
static member ToExpression (e: Expression) = e
就是这样!这现在有效:
> let x = 2.0 + "foo"
Add [Val 2.0; Var "foo"]
> let y = x + "bar"
Add [Add [Val 2.0; Var "foo"]; Var "bar"]
最后,为了减少运行时分配成本,我通常有一个
ToExpressionStub
的单例实例。而不是在每次调用时创建一个新的:
type ToExpressionStub() =
static member val Value = ToExpressionStub()
...
let inline (+) (a: ^a) (b: ^b) = doAdd ToExpressionStub.Value a b
Val
中和
Var
是一个功能,而不是一个错误。程序代码的阅读量远大于其编写量。不要朝自己的脚开枪。
关于generics - 在 F# 中强制内部 DSL 的现有类型,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65395343/
我们有一个系统,其行为由许多 XML 文件定义。(大约 50 个不同的 XML 文件,每个文件管理子系统的行为。) 出于遗留原因,XML 文件采用自定义格式,旨在方便系统中的各种组件使用。 自定义格式
我发送的 json 请求是: Given url applicationURL And path 'applications' And header Authorization = subscribe
默认情况下,生成的 XText 工件会从我的 DSL 生成代码到默认 socket (默认为 src-gen 文件夹)。我知道您可以在 fsa.generateFile("myfile.txt", "
我的论文主题一般是关于领域特定语言的,我想专注于外部或内部 DSL 的设计或实现,但我什至无法思考或开始,因为我在理解 DSL 的概念方面遇到了问题.. 我已经阅读并收集了很多关于这个问题的论文和调查
我有一个 Xtext 项目和几个示例 DSL 文件。我可以使用“示例 Ecore 模型编辑器”打开这些文件并验证它是否符合 Xtext 生成的元模型。但是,由于 DSL 文件未被识别为 Ecore,我
当我学习一些 DSL 时,我意识到 Rebol 中的 Parse 方言可以是一个很好的词法分析器和解析器。 the Parse tutorial 有一个很好的例子: expr: [ter
我正在考虑使用亚马逊云服务(EC2、S3 等)进行托管。我一直在查看可以指定用于配置各种实例的 JSON 元数据,我担心它的复杂性。是否有一个 dsl 可以生成有效的 JSON 元数据并且更重要的是验
我可能会因为这个而被否决,但无论如何我都会试试运气。 我真的找不到 Elastic Search 查询 DSL 的完整形式“DSL”的任何链接,甚至在 Elastic Search 网站上也找不到 h
这个问题可能是复合的,让我扩展一下: 是否存在设计器( stub /框架/元设计器)来创建基于 .NET 对象公共(public) bool 属性的 AND/OR 规则?保存为任何 DSL/Boo/.
与 anko 一样你可以这样写回调函数: alert { title = "" message = "" yesButton { toast("Yes")
我有一个像下面这样的原始聚合脚本,但是很难将其转换为elasticsearch dsl。 我已阅读该文档并找到描述,说我们可以使用.bucket()、. metric()和.pipeline()方法来
如何将这个 gradle groovy 片段转换为 gradle kotlin dsl 而不是非常冗长? test { systemProperties System.getPropertie
这个问题是 ANY operator with jOOQ 的衍生问题和 Are arrays optimized in jOOQ & PostgreSQL? . 我有一个Field field和Lis
我创建了内部 DSL,并且会重载 DefaultGroovyMethods 中的 any() 方法。 class RulesProcessor { } Any live cell with fewer
我正在尝试使用 Gradle 使用 Kotlin 和 Java 11 构建一个简单的 JavaFX 11 程序,按照说明 here .但是,此页面使用 Gradle 的 Groovy DSL,而我正在
如何将这个 gradle groovy 片段转换为 gradle kotlin dsl 而不是非常冗长? test { systemProperties System.getPropertie
我是 Camel 的新手,我仍在学习它,根据我的理解,你可以在 Spring DSL 中做与 Java DSL 相同的事情。我想知道如何将一种转换为另一种?我经常看到 Java 中的例子,但想要 最佳
我正在尝试通过 Groovy 代码为 Jenkins Job DSL 插件创建 ListView 。运行后, View 已创建,但不会添加任何作业。以下代码段之前的代码创建了构建和部署作业,并且在调用
对于 Jenkins 作业 DSL,我试图为作业选择特定的 ssh 代理(插件)键(在包装器上下文中使用 sshAgent 关键字)。我们安装了 Jenkins ssh 代理插件并设置了几个 key
我必须为非程序员(我们公司的客户)创建一个 DSL,它需要提供一些更高级别的语言功能(循环、条件表达式、变量...... - 所以它不仅仅是一个“简单”的 DSL)。 使用 DSL 应该很容易;人们应
我是一名优秀的程序员,十分优秀!