本文以流程控制、运算等为主。说一下go中基本的语法糖。
运算符
go中的运算符和java中的基本一致,但没有++a,--a这种形式的,只有a++,a--这种形式的。其他都一样。
go中比较特殊的一点是,字符串是不能数字直接拼接的。需要采用strconv
包工具类。
示例如下:
//string到int
i, e := strconv.Atoi("33")
//string到int64
int643, e := strconv.ParseInt("233", 10, 64)
fmt.Println(e)
//int到string
string1 := strconv.Itoa(33)
//int64到string
string2 := strconv.FormatInt(99, 10)
fmt.Println(i, int643, e, string1, string2)
可以在下面的提示中看到,所有的入参中,入参第一位是要转换的字符串,入参是base的填10,表示十进制,第三位是bitSize,填写64或者32,表示64或32位。
在返回的参数中,第一位是你需要的值,第二位是异常信息,正常为<nil>,表示无错误。
流程控制
go中的流程控制有以下几种:
- 分支流转类:if、if-else、if-else if、switch、goto、select、defer
- 循环类:for、break、continue
- 异常类:error、panic
if系列
该部分包括所有的if条件语句。go中的if和java中if都是差不多,有一点区别,我们可以从下面的例子中观察一下:
if num := 6; num > 5 {
fmt.Println(num, "大于5")
} else {
fmt.Println(num, "不大于5")
}
可以发现,相较于java,有如下特点:
- if不带(),而且强制带{}(java中一行代码可以不带{}).后面的所有都是如此要求,相对比与java。包括for,switch等。
- if中可以支持对象声明。
for
先来看一个一般的示例:
for num := 0; num < 10; num++ {
fmt.Println(num)
}
这是一个一般输出0到9的例子。
for i, v := range "08,你好!" {
fmt.Println( i,string(v))
}
输出结果:
0 0
1 8
2 ,
3 你
6 好
9 !
前面的i是所处的字节的下标,后面的对应的字符(UTF-8是3个字节)。
注意,此处如果直接输出v,得到的是ASCII值。
goto
goto就是跳转到指定代码段:
fmt.Println("我要执行了")
goto Two
fmt.Println("我是中间代码")
One:
fmt.Println("我是One")
fmt.Println("hhhhh")
goto Third
Two:
fmt.Println("我是two")
goto One
Third:
fmt.Println("我是Third")
输出结果:
我要执行了
我是two
我是One
hhhhh
我是Third
注意:上面的代码如果去掉Third代码段,会造成死循环,因为读完One后会顺序读到Two代码段,而Two又指向One。
select
select和switch类似,但是select是针对于IO的操作。
示例代码:
i1, i2 := 1, 2
c1 := make(chan int, 1)
c2 := make(chan int, 2)
c3 := make(chan int, 3)
go func() { c1 <- i1 }()
select {
case i1 = <-c1:
fmt.Println("接收给", i1, "从c1", c1)
case c2 <- i2:
fmt.Println("发送 ", i2, " 到 c2")
case i3, ok := <-c3:
if ok {
fmt.Println("从c3接收到i3", i3)
} else {
fmt.Println("c3关闭!")
}
default:
fmt.Println("nothing")
}
当某个通道有IO操作时,会执行某条IO后面的代码。如果都有,则随机执行。
上面四种情况:
1)第一个是在i1接收从c1管道发来的变量,是从管道接收值
的操作;
2)第二个是将i2的值发送给c2,是给管道塞值
的操作;
3)从管道中接收值,如果成功,ok为true,否则为false。
4)默认的操作,如果上面都不走,则执行该条操作。如果不设置default,可能会导致程序阻塞,直到有case通过,否则一直阻塞。
defer
defer关键字其后跟随函数或者方法。通过下面的例子来了解一下defer的作用:
示例一
func demo1(path string) ([]byte, error) {
file, err := os.Open(path)
if err != nil {
return nil, err
}
defer file.Close()
return ioutil.ReadAll(file)
}
此处我们用defer来关闭流。
示例二
func demo2() {
f := func(i int) int {
fmt.Println(i)
return i * 10
}
for i := 1; i < 4; i++ {
defer fmt.Println(f(i))
}
}
输出结果为:
1
2
3
30
20
10
可以看到defer没有影响其指定的函数内部的输出,只影响了函数结果的倒序输出。可以得出结论,defer的函数会按顺序进行计算,然后将结果存在栈中,FILO(先进后出),从而输出如上结果。有兴趣的同学可以实验。
示例三
因为我们发现defer和java中的finally有相似之处,都是在代码尾部执行的。但是defer可以写在任何地方,finally有指定的位置。我们需要测试一下defer是否可以改变程序输出的结果。
func demo3()Dog{
dog:=Dog{"ZZ",3}
defer demo31(&dog)
return dog
}
func demo31(dog *Dog) {
dog.age=7
}
输出demo1的结果,发现demo1的结果如下:
{ZZ 3}
得出结论,defer不会影响最终的结果。
总结:defer有以下功能:
- 关闭文件流;
- 使代码最后执行且倒序输出结果;
- 不影响程序最终的结果。
error
在java中,如果出现异常,我们的关键字是Exception。那么在go中,我们的关键字是error。在前面我们也陆陆续续的接触到了error,例如在本节最上面的运算符那里,涉及到了error:
int643, e := strconv.ParseInt("233", 10, 64)
返回的e就是error,如果e为<nil>,则说明一切正常。
还有defer章节的demo1,也是里面的error是error类型。
异常的创建
go中自定义了许多异常,在有些时候,我们需要自定义一些异常。使用的语法是errors.New("...")
。
func main() {
i, e := sqrt(0)
fmt.Println(i, e)
}
func sqrt(num int) (int, error) {
if num == 0 {
return 0, errors.New("The num is invalid!")
}
return num * num, nil
}
go不像java那样去catch异常,它需要你去判断异常是否为nil。
panic
在go中有panic,叫运行时恐慌。他可以层层传递,如果不处理,就会终止程序。
在第二节的通道部分,我们就见识过panic了。
panic是go中的内建函数。在go中有另外一个与panic相对的内建函数——recover()。panic是产生恐慌,而recover是恢复恐慌。而recover函数只能在defer语句中使用,因为defer语句是确定在代码末执行的。此处写一个通用的defer示例:
defer func() {
if p := recover(); p != nil {
fmt.Printf("Fatal error: %s\n", p)
}
}()
下面是手动创建panic并恢复的例子:
func bFunc() {
fmt.Println("Enter bFunc")
panic(errors.New("~~~~~~~~panic~~~~~~~!"))
fmt.Println("Quit bFunc")
}
func aFunc() {
fmt.Println("Enter aFunc")
bFunc()
fmt.Println("Quit aFunc")
}
func main() {
defer func() {
if p := recover(); p != nil {
fmt.Printf("Fatal error: %s\n", p)
}
}()
fmt.Println("enter main")
aFunc()
fmt.Println("out main")
}
输出如下:
enter main
Enter aFunc
Enter bFunc
Fatal error: ~~~~~~~panic~~~~~~!
参考
https://segmentfault.com/a/1190000006815341`
https://www.imooc.com/learn/345