来看一个最简单的go程序:
package main
import (
"fmt"
)
func main() {
fmt.Println("hello world")
}
main函数的左花括号必须在main同行,否则出错。 我刚开始从c/c++转go的时候, 觉得很别扭, 其实, 习惯了就好。 为什么go的先驱们要这么玩呢? 这是为了在语言层面强制让所有使用者统一编程规范, 666.
看看go圣经中的原话, 我来手动打一下: In effect, newlines following certain tokens are converted into semicolons, so where newlines are placed matters to proper parsing of Go code.
什么意思? 假设你把go程序写为:
package main
import (
"fmt"
)
func main()
{
fmt.Println("hello world")
}
编译不过, 为什么呢? 从上面那句英语可知, 上面的程序实际上被转化为:
package main
import (
"fmt"
)
func main();
{
fmt.Println("hello world");
}
显然, main后面多了分号, 这在c/c++中也是不允许的。
现在问题是, 什么时候会自动加入分号呢? 可以看看这个规则(注意上面英语中的certain tokens):
当输入被断开为标记时,如果行末标记为:
a. 一个标识符
b. 一个整数、浮点数、虚数、字符或字符串文字
c. 关键字break、continue、fallthrough或return中的一个
d. 运算符和分隔符++、--、)、]或}中的一个
则分号将被自动插入到标记流中非空白行的末尾
所以, 我们可以知道, 如下go程序是正确的:
package main
import (
"fmt"
)
func add(x, y int) int {
return x + y;
}
func main() {
fmt.Println("main")
i := 3 ; j := 2
fmt.Println(i, j)
add(i,
j)
}
注意, 在调用add函数时, 系统不会在add那一行后面加上分号, 因为此行是以逗号结尾的, 不符合上述规则。
package main
import (
"fmt"
)
func add(x, y int) int {
return x + y;
}
func main() {
fmt.Println("main")
i := 3 ; j := 2
fmt.Println(i, j)
add(i
,j)
}
为什么呢? 在调用add函数时, i作为一个标识符, 被认为是本行结束, i后面被自动加上分号, 所以肯定编译不过。但是, 如果把逗号放到上一行,则OK.
来看看c/c++的情况, 如下程序正常编译正常运行:
#include <iostream>
using namespace std;
int add(int x, int y)
{
return x + y;
}
int main()
{
int sum = add(1
, 2);
cout << sum << endl;
return 0;
}
可见, c/c++表现出不一样的性质, 为什么呢? 因为c/c++不会自动加上分号, 需要程序员自己加, 来标识一个语句的结束。
就这么个逻辑! 依我看, 从c/c++转向go, 经常会手动加上分号, 但习惯后发现, go的方式确实更简洁, 程序员自己不加, go编译器根据规则来加。
最后,大家来看看,这个go程序是否OK呢(注意看花括号的方式)?
package main
import (
"fmt"
)
func main(){
{
x := 123
fmt.Printf("type is %T\n", x)
}
{
x := "123"
fmt.Printf("type is %T\n", x)
}
}
当然OK. 道理不言而喻。