如何在子集类型证明子句中提供非空类型(参数)的任意值?

huangapple go评论89阅读模式
英文:

How to provide an arbitrary value of a non-empty type (parameter) in a subset type witness clause?

问题

I understand your code-related query. Here's the translated code portion:

我正在尝试引入一个参数化的子类型(`Bar<T(0)>` 如下),需要我提供类型 `T` 的证明:
```dafny
datatype Foo<T> = Foo(x: T)
type Bar<T(0)> = x: Foo<T> | true witness ???

由于 T 是非空的,我不明白为什么这不可能。

我的尝试:

type Bar<T(0)> = x: Foo<T> | true witness Foo(*) // 错误:需要关闭括号
type Bar<T(0)> = x: Foo<T> | true witness Foo(var x: T := *; x) // 错误:无效的 UnaryExpression
type Bar<T(0)> = x: Foo<T> | true witness Foo(var x: T :| true; x) // 错误:为了可编译,let-such-that 表达式的值必须唯一确定

我猜想我可能可以编写一个函数、函数方法或方法来帮助,但在这里也失败了。一个方法可以用来获得任意的 T

method Pick<T(0)>() returns (x: T)
{
}

但是 witness 子句需要一个函数方法,而 ghost witness 需要一个函数。在尝试在这两者中获取任意的 T 时,我得到了与我上面指出的错误相同的错误。在方法或函数方法中调用 Pick 也是不可能的。

我明白我可以指定 witness *,但这似乎更像是一个权宜之计而不是一个合适的解决方案。或者我不应该为此烦恼吗?


Please note that the "???" in the code represents the part where you were encountering difficulties, and it hasn't been resolved in the translation.

<details>
<summary>英文:</summary>

I am trying to introduce a parameterized subset type (`Bar&lt;T(0)&gt;` below) requiring me to provide a witness of type `T`:
```dafny
datatype Foo&lt;T&gt; = Foo(x: T)
type Bar&lt;T(0)&gt; = x: Foo&lt;T&gt; | true witness ???

Since T is non-empty, I don't see a reason why this should not be possible.

My attempts:

type Bar&lt;T(0)&gt; = x: Foo&lt;T&gt; | true witness Foo(*) // Error: closeparen expected
type Bar&lt;T(0)&gt; = x: Foo&lt;T&gt; | true witness Foo(var x: T := *; x) // Error: invalid UnaryExpression
type Bar&lt;T(0)&gt; = x: Foo&lt;T&gt; | true witness Foo(var x: T :| true; x) // Error: to be compilable, the value of a let-such-that expression must be uniquely determined

I speculated that I may be able to write a function, function method or method to help out, but failed here as well. A method can be used to obtain an arbitrary T:

method Pick&lt;T(0)&gt;() returns (x: T)
{
}

But a witness clause would require a function method while a ghost witness would require a function. When trying to obtain an arbitrary T in either of those, I get the same errors as the ones I pointed out above. Calling Pick in methods or function methods isn't possible either.

I understand that I can specify witness *, but this seems more of a hack than a proper solution. Or should I not be bothered by it?

答案1

得分: 2

@DivyanshuRanjan 在一个评论中回答了这个问题,我将尝试在这里进行总结。

以下内容有效:

type Bar&lt;T(0)&gt; = x: Foo&lt;T&gt; | true ghost witness Foo(var x: T :| true; x)

它有效是因为ghost见证不需要可编译,并且var x: T :| true; x是一个在函数中有效的表达式,即非编译的上下文。在非ghost见证子句中,验证器将拒绝此表达式并显示上述错误("错误:为了可编译,let-such-that表达式的值必须唯一确定")。

根据参考手册ghost witness的缺点是"[Dafny验证器]将无法在编译的代码中自动初始化该类型的变量"。

以下内容无效:

type Bar&lt;T(0)&gt; = x: Foo&lt;T&gt; | true ghost witness Foo(var x: T := *; x)

根据参考手册*是"右侧表达式":
>右侧表达式是一种类似表达式的构造,可能具有副作用。因此,这种表达式只能在方法的某些语句中使用,而不能作为一般表达式或在函数或规范中使用。RHS要么是数组分配,要么是对象分配,要么是破坏右侧,要么是方法调用,要么是简单表达式,可选地后跟一个或多个属性。

我看不出var x: T := *; xvar x: T :| true; x有何不同(除了一个可在函数中使用,另一个不行)。希望将来可以改进这部分答案。

英文:

@DivyanshuRanjan answered the question in a comment, which I will try to summarize here.

The following works:

type Bar&lt;T(0)&gt; = x: Foo&lt;T&gt; | true ghost witness Foo(var x: T :| true; x)

It works because ghost witnesses don't need to be compilable and var x: T :| true; x is an expression that is valid in functions, i.e. non-compiled contexts. In a non-ghost witness clause the verifier would reject this expression with the error shown above ("Error: to be compilable, the value of a let-such-that expression must be uniquely determined").

According to the reference manual, the drawback of a ghost witness is that "[the Dafny verifier] will not be able to auto-initialize a variable of that type in compiled code."

The following is not valid:

type Bar&lt;T(0)&gt; = x: Foo&lt;T&gt; | true ghost witness Foo(var x: T := *; x)

According to the reference manual * is a "Right-Hand-Side Expression":
>A Right-Hand-Side expression is an expression-like construct that may have side-effects. Consequently such expressions can only be used within certain statements within methods, and not as general expressions or within functions or specifications.
An RHS is either an array allocation, an object allocation, a havoc right-hand-side, a method call, or a simple expression, optionally followed by one or more attributes.

I fail to see how var x: T := *; x differs from var x: T :| true; x (other than the fact that one is usable in functions and the other isn't.) I hope that this part of the answer can be improved in the future.

huangapple
  • 本文由 发表于 2023年5月6日 19:47:06
  • 转载请务必保留本文链接:https://go.coder-hub.com/76188695.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定