坐忘峰 golang入坑系列

读前必读: 本文写于20日,首发于gitbook. 迟到的是日期,没变的是内容。 点击进入 https://andy-zhangtao.gitbooks.io/golang/content/ 可以看到最新内容(并非广告,只是因为博客园的排版太难看了)。

告别昨日梦,今朝艳阳天。

大道通四方,车影舞翩翩。

不做旁观者,去向应了然。

白云绕远树,飞燕春汛传。

天下万千景,彩绘壮河山。

我辈当努力,敢把宏图展。

迈步似虎跃,挥臂龙盘旋。

天地更广阔,豪气入云端。

送给刚度过周末,又要开始上班的程序员们。按照惯例,每周的周一都应该是复盘上周进度,制定本周计划的时间点。每每到了这一天,都是几家欢喜几家愁,有的庆幸进度正常,有的发愁扯皮何时休。欢喜也罢,发愁也罢,在这个天干物燥的冬季,还是要保持乐观平和的心态。气大伤身,提防感冒。老人说得好,酒是穿肠毒药,色是刮骨钢刀, 财是惹祸根苗,气是无烟火炮。这世上吧,没有无缘无故的夸奖,也没有无缘无故的批评。轮到自己挨批评了,别真真计较,心怀宽广,莞尔一笑,世界就变得美好了许多。

话说太多就是烦, 本文也不是心理访谈,所以就此打住不再延伸。总之一句话,工作就是一个工作,又不是你的全部,干嘛那么较真。真要较真,咱们就应该较真一下Golang 。 毕竟,我是写技术文章的,技术才是老本行。 开导人生,那是鞠萍姐姐的工作,下面才是我的工作。

Golang的判断

golang的判断也就是If..else.. , switch和select。 通俗点就是如果满足条件去干嘛,如果不满足条件去干嘛。每种编程语言都有这么一套判断机制,而且判断的标准也大同小异。而唯一不同的地方,就是语法细节千奇百怪。 大胆猜测一下,可能目前流行的编程语言都或多或少受到了C语言的影响,而且绝大多数的程序员开山之课都是C语言,也就造成先入为主的影响。因此C语言是什么风格,新的语言就要仿造的八九不离十,要不学习曲线太陡,光吓就要吓走一大波人。

所以,如果你有其它语言的学习基础,本节基本可以一扫而过。 如果你没有其它语言基础,或者对自己没有足够的自信,就看一看吧。 技多不压身,多学点没坏处。

先来看If。

If是典型的判断语句。 语法是:

If <条件> {}

例如:
If a >= 0 {
    xxxxx
}

在条件中括号不是必须存在的,可加可不加。 如果你从JS转过来学Golang,那么恭喜你了,加上也没事。 如果Golang转过去学Js,每次都要记得添加括号,这个习惯不是一会半会能扳过来的。

然后再看If ... else ...

上面是如果满足怎么办,属于单元判断。 而If ... else... 是双元判断,如果满足怎么办,否则怎么办.

If <条件>{
    满足的情况
} else{
    不满足的情况
}

双元+单元=多元。将If .. else ..级联起来,就是多元判断。例如:

if <条件1> {

} else if <条件2> {

} else if <条件3> {

}

同样是多元判断,switch的语法看起来就会比If .. else if 要简洁。

switch <表达式>{
    case <值1>:
    case <值2>:
    defalut:
}

上面说过,每个语言在细节上面都有不同。 在Golang中:

  • switch判断不需要添加break。别问为什么,编译器就是这么规定的,匹配一个后,就会退出这个语义块,所以不需要break
  • default是可选项,同样不需要问为什么,如果都匹配不成功,那就是匹配不成功。没有异常,没有报错。所以有可能程序出现非预期结果。
  • 表达式部分必须是一个完整的布尔表达式,或者是一个推算出唯一结果的函数。 如果表达式为空,或者计算失败,则默认为True。
  • 所有的case必须是同一种数据类型,如果类型不一致,就有可能出现非预期的计算结果
  • 通过fallthrough可以跳过case判断。例如:
a := 1
switch {
    case a >= 1:
        fmt.Println("a>=1")
        fallthrough
    case a >= 2:
        fmt.Println("a>=2")
        fallthrough
    case a >= 3:
        fmt.Println("a>=3")
        fallthrough
    case a >= 4:
        fmt.Println("a>=4")
        fallthrough
    default:
    }

执行后的结果是:

a>=1
a>=2
a>=3
a>=4

因为满足条件一后,执行fallthrough,则跳过case2的判断直接执行case2的语句,后面依次类推,因为在case4时也存在fallthrough,所以必须存在defalut,要不然语义解析失败。如果case4没有fallthrough,那么default就可以删除。

switch除了这种用法之外,在生产环境中,还经常用来判断值类型,例如:

switch x.(type){
   case type:
      statement(s);      
   case type:
      statement(s); 
   default: 
      statement(s);
}

将表达式替换为x类型断言,如果是int型则怎么办? 如果是string类型则怎么办? 这种用法经常出现在接口中,当可能返回多种类型时,通过这种方式来逐一判断处理。下面举一个例子来说明问题:

var x interface{}

   switch i := x.(type) {
      case nil:      
         fmt.Printf("type of x :%T",i)                
      case int:      
         fmt.Printf("x is int")                       
      case float64:
         fmt.Printf("x is float64")           
      case func(int) float64:
         fmt.Printf("x is func(int)")                      
      case bool, string:
         fmt.Printf("x is bool or string")       
      default:
         fmt.Printf("don't know the type")     
   }
在Golang判断环节,还有经常使用的一种判断: Select

select语法和switch语法非常相像,有些初学者很容易将两者弄混。但除了样子相近之外,二者作用完全不同。先看外表:

select {
   case communication clause  :
      statement(s);      
   case communication clause  :
      statement(s); 
   default :
      statement(s);
}

如果把select换成switch,活脱脱一个条件判断表达式。 所以外表极其相似。但你要注意:

  • select后面没有表达式
  • case后面的类型必须是chan。(chan我们还没有提到,提前剧透一下,chan是一种数据通信管道,golang封装成了数据类型)
  • default是可选的。如果有default,必须放到select最后。

因为我们没有讲到chan,所以目前来说记住上面的部分就可以了。

Golang的判断相对于其它语言来说,只有If, switch和select三种。 其中select还属于特殊的判断。所以只剩下if和switch两种。相当好记,相当好用。

如果本节就这么结束了,不用你说,我都感觉内容不详实(不详实也不是一次两次了,这次是由衷的感觉不详实)。所以再写一下Golang里面的循环。放心内容不会多的,我就想写成口袋书,一次就看一部分,不用长篇累牍的写那么多。 你们看着累,我写的也累。

Golang的循环只有For循环

好了,写完了。 板砖尽情的扔过来吧,让我来感受一下遭人抨击的感觉吧。 虽说你们扔了板砖,但我仍然要说Golang只有For循环一种循环语法。 没有什么while循环,do..for循环,foreach循环,....map()循环之类的东西。在Golang当中需要使用循环,记住只有For循环。

for [condition |  ( init; condition; increment ) | Range] {
   statement(s);
}

上面就是For循环的三种写法。

  • condition。 条件判断。 也就是传说中的 for true{} 循环,某种意义上等于与while循环
  • init; condition; increment。初始化循环,预定俗称的写法就是初始化i:=0,然后当i<=N时循环,每次i++。
  • Range,迭代循环。基本用来迭代数组.例如;
numbers := [6]int{1, 2, 3, 5} 
for i,x:= range numbers {
      fmt.Printf("value of x = %d at %d\n", x,i)
}

下面用一个例子,来演示三种情况:

   var b int = 15
   var a int
   numbers := [6]int{1, 2, 3, 5} 

   /* for loop execution */
   for a := 0; a < 10; a++ {
      fmt.Printf("value of a: %d\n", a)
   }
   for a < b {
      a++
      fmt.Printf("value of a: %d\n", a)
   }
   for i,x:= range numbers {
      fmt.Printf("value of x = %d at %d\n", x,i)
   }

如果一个For循环使用的不过瘾,可以嵌套多个For循环。直接看语法:

for [condition |  ( init; condition; increment ) | Range] {
   for [condition |  ( init; condition; increment ) | Range] {
      statement(s);
   }
   statement(s);
}

语法看不懂,那就给你一个实例,比对着看吧:

   var i, j int

   for i = 2; i < 100; i++ {
      for j = 2; j <= (i/j); j++ {
         if(i%j==0) {
            break; // if factor found, not prime
         }
      }
      if(j > (i/j)) {
         fmt.Printf("%d is prime\n", i);
      }
   }

上面实例中,出现了一个break。这是标准的退出循环的方式。 那么Golang中有几种退出循环的方式呢?

  • Break。 使用率最高的方式,当满足退出条件时,直接break退出
  • Continue。有人说不算出退出循环,因为它只是跳出了本次循环,继续下次循环。但如果不放这里,我又把continue放哪里呢?所以暂且就把continue当做跳出循环吧。只不过是跳出本次循环而已。
  • Goto。 武林界失传已久的Goto大法,在Golang中重出江湖。 与其它语言,强制半强制的屏蔽Goto不同,golang很大方的推出goto。没有什么其它限制条件,只要你认为goto使用的正确就可以大胆的使用。(小心陷入死循环)。使用Goto时,需要提前定义标签,然后goto 标签即可。 下面是一个实例:
   var a int = 10

   LOOP: for a < 20 {
      if a == 15 {
         a = a + 1
         goto LOOP
      }
      fmt.Printf("value of a: %d\n", a)
      a++     
   }

如果你认真的分析一下代码,或者自己执行一下代码就会发现:你死循环了!所以不限制用,不等于滥用。 不是老司机,就别调试小寡妇。 寡妇不是病,真闹起来,要你命。所以用之前,慎重考虑一下是否真的要用。 记住:no zuo no die!

侯门一入深如海, 从此萧郎是路人。 代码之路漫漫而长,如果从头开始一直在跟着本文的更新,那么你现在已经算是踏入了Golang的大门,至少我还没有开始摧残你那颗幼小的心灵。 由易而难,慢慢来。 跟着练,悄然之间你就会发现Golang原来就是这么一回事。

你可能感兴趣的:(坐忘峰 golang入坑系列)