[Go]条件语句

 1 package main
 2 
 3 import (
 4     "io/ioutil"
 5     "fmt"
 6 )
 7 
 8 //条件语句
 9 //if的条件语句不需要括号
10 //if的条件里可以赋值,if的条件里赋值的变量作用域就在这个if语句里
11 //switch会自动break,除非使用fallthrough
12 
13 func grade( score int ) string {
14     g := ""
15     switch  {  //switch里面可以没有表达式
16     case score < 0 || score >100:
17         panic(fmt.Sprintf( "Wrong score: %d", score) )  //中断程序执行,报错
18     case score < 60:
19         g = "F"
20     case score < 80:
21         g = "C"
22     case score < 90:
23         g ="B"
24     case score <= 100:
25         g ="A"
26     }
27     return g
28 }
29 
30 func main() {
31     const filename = "abc.txt"
32 /*  contents, err := ioutil.ReadFile(filename)
33     if err != nil {
34         fmt.Println(err)
35     }else {
36         fmt.Printf("%s\n", contents)
37     }*/
38     //或者也可以用写成下面这样
39      if contents, err := ioutil.ReadFile(filename); err != nil{   //contents和err 定义在if里,生存期只在if里面才有
40          fmt.Println(err)  //open abc.txt: The system cannot find the file specified.
41      }else {
42          fmt.Printf("%s\n", contents)
43      }
44 
45      fmt.Println(
46          grade( 0 ),  //F
47          grade( 59),  //F
48          grade( 99),  //A
49          grade(100),  //A
50      )
51 }

 

上面介绍了switch语句中switch表达式和case表达之间有着怎样的联系?

  switch表达式的结果值与某个case表达式中任意一个子表达式的结果值相等,该case表达式所属的case子句就会被选中,并且一旦某个case子句被选中,其中的附带在case表达式后边的那些语句就会被执行,同时其他的所以case子句都会被忽略(如果被选中的Case子句附带的语句列表中包含了fallthrough语句,那么紧挨在它下边的那个case子句附带的语句也会被执行) 

  正因为存在判断相等的操作,switch语句对switch表达式的结果类型,以及各个Case表达式中子表达式的结果类型都是有要求的(Go语言中,只有类型相同的值之间才有可能被允许进行判等操作)

    value1 := [...]int8{0, 1, 2, 3, 4, 5, 6}
    switch 1 + 3 { // 这条语句无法编译通过。
    case value1[0], value1[1]:
        fmt.Println("0 or 1")
    case value1[2], value1[3]:
        fmt.Println("2 or 3")
    case value1[4], value1[5], value1[6]:
        fmt.Println("4 or 5 or 6")
    }

  上面的代码中,1+3的求值结果是无类型的常量4,这个常量会被自动地转换为此种常量的默认类型的值,比如整数4的默认类型是int,浮点数3.14的默认类型是float64

  因此上面的switch表达式的结果类型是int,而那些case子表达式的结果类型却是int8,它们类型不同,是无法通过编译的

  再看下面这段代码的switch表达式的结果值是int8类型的,那些case表达式中的结果值是无类型的常量,却是可以编译的

    value2 := [...]int8{0, 1, 2, 3, 4, 5, 6}
    switch value2[4] {
    case 0, 1:
        fmt.Println("0 or 1")
    case 2, 3:
        fmt.Println("2 or 3")
    case 4, 5, 6:
        fmt.Println("4 or 5 or 6")

  这是因为如果case表达式中子表达式的结果值是无类型的常量,那么它的类型会被自动地转换为swicth表达式的结果类型,又由于上述那几个整数都可以被转换为int8类型的值么,所以对这些表达式的结果值进行判等操作是没有问题的

  即Swicth语句会进行有限的类型转换,但不能保证这种转换可以统一它们的类型

 

switch语句对它的case表达式还有哪些约束?

  1)switch语句不允许case表达式中的子表达式结果值存在相等的情况

    value3 := [...]int8{0, 1, 2, 3, 4, 5, 6}
    switch value3[4] { // 这条语句无法编译通过。
    case 0, 1, 2:
        fmt.Println("0 or 1 or 2")
    case 2, 3, 4:
        fmt.Println("2 or 3 or 4")
    case 4, 5, 6:
        fmt.Println("4 or 5 or 6")
    }

  但是这个约束只针对结果值为常量的子表达式,比如1+1和2不能同时出现。但是也就可以想办法绕过这个对子表达式的限制了,下面的代码就可以绕过限制

    value5 := [...]int8{0, 1, 2, 3, 4, 5, 6}
    switch value5[4] {
    case value5[0], value5[1], value5[2]:
        fmt.Println("0 or 1 or 2")
    case value5[2], value5[3], value5[4]:
        fmt.Println("2 or 3 or 4")
    case value5[4], value5[5], value5[6]:
        fmt.Println("4 or 5 or 6")
    }

  但是上面这种绕过方式对用于类型判断的switch语句无效,因为类型switch语句中的case表达式的子表达式必须直接由类型字面量表示,而无法通过间接的方式表示。

    value6 := interface{}(byte(127))
    switch t := value6.(type) { // 这条语句无法编译通过。
    case uint8, uint16:
        fmt.Println("uint8 or uint16")
    case byte:
        fmt.Printf("byte")
    default:
        fmt.Printf("unsupported type: %T", t)
    }

  这里byte是uint8类型的别名类型,它们本质上是同一个类型,只是类型名称不同,因此是无法通过编译的

 

posted @ 2018-07-13 18:25  yuxiaoba  阅读(191)  评论(0编辑  收藏  举报