go - 如何将函数参数静态限制为值的子集

标签 go static-typing

如何将函数参数静态约束为所需类型的值子集?

值集将是在包中定义的一个小集合。最好是编译时检查而不是运行时检查。

我能想到的唯一方法是这样的:

package foo

// subset of values
const A = foo_val(0)
const B = foo_val(1)
const C = foo_val(2)

// local interface used for constraint
type foo_iface interface {
    get_foo() foo_val
}

// type that implements the foo_iface interface
type foo_val int
func (self foo_val) get_foo() foo_val {
    return self
}

// function that requires A, B or C
func Bar(val foo_iface) {
    // do something with `val` knowing it must be A, B or C
}

因此,现在包的用户无法用任何其他值代替 ABC

package main

import "foo"

func main() {
    foo.Bar(foo.A) // OK
    foo.Bar(4)     // compile-time error
}

但是完成这个看似简单的任务似乎需要很多代码。我有一种感觉,我把事情搞得太复杂了,错过了语言中的一些功能。

该语言是否具有某些可以用简洁的语法完成相同任务的功能?

最佳答案

Go 做不到这一点(我不认为,我不认为几个月让我有经验)

ADA 可以,C++ 有时可以,但不是很干净(constexpr 和 static_assert)。

但真正的问题/要点在这里,为什么重要?我玩 Go 和 GCC 作为编译器,GCC 真的很聪明,尤其是 LTO,持续传播是最容易应用的优化之一,它不会打扰检查(你是(我们在 C 中调用的)静态初始化 A B 和 C,GCC 对此进行优化(如果它有函数的定义,那么它有 LTO))

现在这有点跑题了,所以我将停止讨论那个混合的 blob,但是除非您的程序受 CPU 限制,否则测试值的合理性是很好的。别担心。

总是写一些更容易阅读的东西,你会感谢你后来所做的

你的运行时检查也是如此,如果编译器有足够的信息可以提供,如果它可以推断(证明)它们不会抛出,它就不会费心去做它们,因为它会很容易地发现它的常量值。

附录

很难进行编译时检查,例如 c++ 中的 constexpr 非常有限(它涉及的所有内容也必须是 constexpr 等)- 它不能很好地与普通代码一起使用。

假设一个值来自用户输入?该检查必须在运行时进行,如果您编写了两组约束(无论如何可行),那将是愚蠢的(并且违反了 DRY),一组用于编译,一组用于运行。

我们能做的最好的事情就是让编译器变得非常聪明,而 GCC 就是。我敢肯定其他人也很好(除了 MS,我从来没有听到过对它的赞美,但是作者很聪明,因为他们一开始就写了一个 C++ 解析器!)

关于go - 如何将函数参数静态限制为值的子集,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20527226/

相关文章:

java - 仅允许 null 选择加入

java - 为什么java排名第一?

ruby - 是什么阻止了静态类型语言拥有像 Ruby 的 method_missing 这样的东西?

python - 如何在 Cython 中有效地使用 Python 风格的整数?

json.Marshal 与 Encoder.Encode

http - 是否可以从 http Handle 函数写入 channel ?

go - Go 中令人困惑的并发和性能问题

c - 戈朗 : How to correctly parse UTF-8 string from C

postgresql - `gorm` 忽略 `sql:"索引 "` 标签

haskell - 是否有用于更新嵌套数据结构的 Haskell 习惯用法?