golang并没有像c语言一样提供三元表达式。三元表达式的好处是可以用一行代码解决原本需要多行代码才能完成的功能,让冗长的代码瞬间变得简洁。不过对于新手来说,建议还是少用三元表达式。在这里,我用golang通过函数的方式实现了三元表达式。
官方faq推荐的做法是用 if 代替:
不过用 if 的问题是变量 n 有作用域问题. 我们需要在 if 之前先定义变量 n,这样才可以在 if 语句之后使用变量 n。
本来一个简单的 n := expr? trueval: falseval 就能够表达的问题,变的复杂了很多。 这和go所追求的简单思路是有冲突的。
类似的有 max/min 等函数。因为这类函数使用频度比较高,在很多pkg的内部都定义了私有的实现。
熟悉go语言的用户应该可以发现,这个 max 只支持 int 类型。 对于支持泛型的c++语言来说,max 一般被实现为一个模板函数:
在c++版本中,不仅用到的泛型t,还依赖 a<b 的运算符重载特性。 在c语言中,虽然没有泛型和运算符重载,但是三元表达式也具备全部的特性(因为表达式天生就是支持泛型的)。
而这些都是go语言中缺少的特性。 不过在go语言中可以模拟一个更普通的函数(if 的首字母大写,是函数名,不是 if 关键字):
有几个关键点:
- go不支持运算符重载,因此需要先将 a<b 在函数外转换为 bool 条件
- go不支持泛型,只能用 interface{} 模拟
- 返回的类型安全需要用户自己保证,.(type) 的类型必须匹配
- interface{} 是运行时泛型,性能没有编译时泛型高
由此可见,?: 不仅仅是一个简单的三元表达式。其实它更像一个内置的泛型版的函数(因为表达式天生就是支持泛型的)。
期望未来的go版本中,能完善对 ?: 三元表达式 和 编译时的泛型 的支持。
补充: 星星 同学的提示:可能会导致深入嵌套的滥用: c?d?e?0:1:2:3 。 因为三元表达式是一个表达式,必然是允许嵌套的。
不过我觉得嵌套不是问题的本质,函数也能导致嵌套的滥用。 但是不能因为滥用的行为来排斥有存在价值的语法(比如三元表达式)。
示例
执行结果