zoukankan      html  css  js  c++  java
  • golang笔记——流程控制

    条件语句

      if ... else if ... else 语句,如:

    if num > 100{
        fmt.Println(">100")    
    } else if 0 < num {
        fmt.Println("<0")
    } else {
        fmt.Println("0<<num<<100")
    }

      惯用法:if 语句条件前面可以前置一条初始化语句,而go语言中的函数可以返回多个值,有很多函数第二个返回值是 error 类型,则我们可以通过

    if v, err := func(); err == nil {
      //逻辑语句
    }

    循环语句

      GO语言中的 switch 语句分为两种,一种是表达式 switch 语句,和其它语言中的 switch 使用方法相同;另一种是类型判断 switch 语句,它类似于类型断言,但使用 type 关键字来充当被判断的类型。

      表达式 switch 示例如下:

        //表达式 switch
        switch content := getContent(); content {
        default:
            fmt.Println("Unknow language")
        case "Lua":
            break
        case "Python":
            fmt.Println("python")
        case "C", "C++", "Java":
            fmt.Println("A compiled language")
        }

      类型判断 switch 语句示例如下:

        v := "3"
        switch interface{}(v).(type) {
        case string:
            log.Printf("Thie string is '%s'.
    ", v)
        case int, uint, int8, uint8, int16, uint16:
            log.Printf("Thie integer is %d.
    ", v)
        default:
            log.Printf("Unsupported value.(type=%T)
    ", v)
        }

      此外,switch 语句还可以实现串联 if 语句的替代方案,可以使代码看起来更清晰易读,在 switch 表达式缺失的情况下, switch 判定目标会被视为布尔类型,第一个返回 true 的 case 表达式将会被执行,如:

        switch {
        case num > 100:
            log.Println(">100")
        case num < 0:
            log.Println("<0")
        default:
            log.Println("0<<num<<100")
        }

      再此外,switch 的 case 语句最后一行可以加上 fallthrough,表示继续执行下一个case(不需要匹配条件表达式),这个有什么作用呢?猜想可能在某个顺序工作流中,比如根据一个status的值,依次执行某些动作,如果每个case最后都有这个 fallthrough,那么无论当前 status 是多少,都能保证执行完剩下的动作。注意,fallthrough必须要放在 case块的结尾,且如果前面有 break,将不会执行。(break就提前结束了本次case了,这个可没有 defer 的效果)

      for语句有三种用法,一是常规用法,结构先后是初始化子句、条件、后置子句,如:

        sum := 0
        for i := 0; i < 100; i++ {
            sum += i
        }

      二是类似其它语言中 while 的作用,注意GO语言中没有while语句,如:

        i := 0
        for i < 100 {
            i += 2
        }

    而不使用任何条件则表示死循环:

    for {
        //...    
    } 

     三是类似其它语言中 foreach 的作用,用来迭代string切片字典等类型,如:

        m := map[string]int{"A": 1, "B": 2}
        for k, _ := range m {
            log.Print(k)
        }

      注意:if/for/switch 语句都可以接受一个可选的初始化子句; breakcontinuegoto 语句都可以跳转到指定标记,标记的定义使用 "标识符:" 的形式。

      for ... range 语句本身是只读的,如下面的代码是不会改变原数据的:

        m := make(map[string]int)
        m["a"] = 1 
        m["b"] = 2 
        for _, v := range m { 
            v = v * 10
        }   
        for _, v := range m { 
            println(v)
        }   
    
        s := []int{1, 2}
        s[0] = 1 
        s[1] = 2 
        for _, v := range s { 
            v = v * 10
        }   
        for _, v := range s { 
            println(v)
        }   

      但是,如果 v 本身就是指针,那自然还是能修改的,或者对于切片和字典来说,通过 for ... range 的索引或键来修改都是可行的。

      也可以对一个函数进行 range (当然前提是这个函数返回值是可 range的类型,如字典切片等),函数只会被调用一次:

    for userid, user range manager.GetUserManager().GetOnlineUserMap {
        println(userid)
    }

    goto语句

      goto语句只能配合标记来执行,跳转到指定位置,该语句在其它语言中很有争议,一般为了代码可读性都不推荐使用。

    defer语句

      GO语言特有的一个流程控制语句,它用来预定对一个函数的调用。它只能出现在一个函数中(假设是A函数),且只能调用另一个函数(假设是B函数),意味着在A函数结束返回时,延迟调用B函数,一般用于打开文件时的资源清理等工作。如果一个函数内部调用多个 defer 语句,则遵循后进先出的原则。defer 语句后面可以跟着匿名函数,来快速实现一些临时的功能。defer 调用的函数可以使用的变量,可以是通过参数传进来的,也可以是上下文中可以调用的变量,如果是传参进来的,则会立即被求值,如果是上下文中的变量,则不会立即被求值,而是取在 defer 函数调用时的值,这一点要注意。

    异常处理语句

      系统提供了一个 error 接口,定义如下:

    type error interface {
           Error() string    
    }

      GO中习惯使用 error 类型值来表明非正常的状态,但我们不需要自己去创建一个实现 error 接口的类型,而只需要通过 errors 包提供的 New 方法来创建,如: errors.New("this is error");  我们来看一下 errors 的源代码:

    package errors
    
    func New(text string) error {
        return &errorString{text}
    }
    
    type errorString struct {
        s string
    }
    
    func (e *errorString) Error() string {
        return e.s
    }

      可以直接把 error 类型值传递给 fmt.Print 方法参数,会自动检测并输出 error 类型值的 Error() 方法。

      除了使用 erros.New 方法创建 error 类型值外,我们还可以使用 fmt.Errorf 函数来创建,它适合创建格式化字符串形式的 error类型值,注意这个方法并不会打印输出到屏幕,内部还是调用 errors.New 来实现的,如:

    err := fmt.Errorf("%s
    ", "nil error!")

      除了上面两种简便的创建 error 类型值的方法之外,我们也可以通过自定义实现 error 接口的方式来创建,比如 os.PathError 就是一个 error 接口的实现类型。查看 os/error.go 的源代码:

    type PathError struct {
        Op   string
        Path string
        Err  error
    }
    
    func (e *PathError) Error() string { return e.Op + " " + e.Path + ": " + e.Err.Error() }

      当遇到不可恢复的错误状态时,我们使用 panic 与 recover 来处理异常,这个异常就类似于其它语言中的异常了,而 error 相对而言只能算是一种状态码。

      panic 接受一个做任意类型的函数(通常是 string 或 error 类型),然后停止当前的控制流程,将控制权交给调用它的函数,但调用它的函数的执行也将被停止,再继续向上传播。GO语言中使用 recover 来捕获这样的异常,它可以使当前的程序从异常状态中恢复并重新获得流程控制权,并返回 interface{} 类型。通常,我们在 defer 语句中调用一个匿名的函数,来进行 recover 处理(因为异常时虽然流程已经不可控,但 GO 保证 defer 语句会执行),可以通过 recover() 获取当前的异常,如果不为 nil,表示存在异常,且如果该异常是由GO语言运行时程序引起的,返回的将是 runtime.Error 类型的值。如:

        defer func() {
            if r := recover(); r != nil {
                fmt.Printf("Recoverd panic:%s
    ", r)
            }
        }()

      如果一个函数里有多个 defer 语句,注意其执行顺序遵循先进后出的原则。

  • 相关阅读:
    设置Jmeter默认中文页面
    spring boot controller设置 @Transactional 不回滚的解决办法
    SpringBoot事务注解@Transactional
    spring boot @Transactional事物处理
    Spring事务管理中@Transactional
    @Transactional 注解的使用和注意
    spring的@Transactional注解详细用法
    【Developer Log】ProGuard扰码可执行JAR包
    使用Jprofiler+jmeter进行JVM性能调优
    SQL中如何使用EXISTS替代IN
  • 原文地址:https://www.cnblogs.com/tianyajuanke/p/5217601.html
Copyright © 2011-2022 走看看