巨细靡遗流程控制,Go lang1.18入门精炼教程,由白丁入鸿儒,Go lang流程结构详解EP09

简介: 流程结构就是指程序逻辑到底怎么执行,进而言之,程序执行逻辑的顺序。众所周知,程序整体都是自上由下执行的,但有的时候,又不仅仅是从上往下执行那么简单,大体上,Go lang程序的流程控制结构一共有三种:顺序结构,选择结构,循环结构。顺序结构:从上向下,逐行执行;选择结构:条件满足,某些代码才会执行,0-1次;循环结构:条件满足,某些代码会被反复的执行多次,0-N次

流程结构就是指程序逻辑到底怎么执行,进而言之,程序执行逻辑的顺序。众所周知,程序整体都是自上由下执行的,但有的时候,又不仅仅是从上往下执行那么简单,大体上,Go lang程序的流程控制结构一共有三种:顺序结构,选择结构,循环结构。顺序结构:从上向下,逐行执行;选择结构:条件满足,某些代码才会执行,0-1次;循环结构:条件满足,某些代码会被反复的执行多次,0-N次

选择结构之条件判断if/else

市面上的语言都有if/else逻辑,逻辑非常简单,只要满足条件,就会执行条件代码块的逻辑:

if 布尔表达式 {  
   /* 在布尔表达式为 true 时执行 */  
}  
  
if 布尔表达式 {  
   /* 在布尔表达式为 true 时执行 */  
} else {  
  /* 在布尔表达式为 false 时执行 */  
}  
  
if 布尔表达式1 {  
   /* 在布尔表达式1为 true 时执行 */  
} else if 布尔表达式2{  
   /* 在布尔表达式1为 false ,布尔表达式2为true时执行 */  
} else{  
   /* 在上面两个布尔表达式都为false时,执行*/  
}

具体逻辑:

package main  
  
import "fmt"  
  
func main() {  
  
    var a int = 1  
  
    /* 使用 if 语句判断布尔表达式 */  
    if a < 20 {  
        /* 如果条件为 true 则执行以下语句 */  
        fmt.Printf("a 小于 20\n")  
    }  
    fmt.Printf("a 的值为 : %d\n", a)  
}

程序返回:

a 小于 20  
a 的值为 : 1

需要注意的是,条件变量类型要一致才能比较。

除此之外,if还可以在判断之前执行逻辑:

package main  
  
import (  
    "fmt"  
)  
  
func main() {  
    if num := 10; num%2 == 0 { //checks if number is even  
        fmt.Println(num, "is even")  
    } else {  
        fmt.Println(num, "is odd")  
    }  
} 

程序返回:

10 is even

也就是说,当判断变量num对2取余是否等于0之前,我们可以先给num进行赋值操作。

同时if/else也支持多分支判断:



package main  
  
import (  
    "fmt"  
)  
  
func main() {  
    score := 88  
    if score >= 90 {  
        fmt.Println("成绩等级为A")  
    } else if score >= 80 {  
        fmt.Println("成绩等级为B")  
    } else if score >= 70 {  
        fmt.Println("成绩等级为C")  
    } else if score >= 60 {  
        fmt.Println("成绩等级为D")  
    } else {  
        fmt.Println("成绩等级为E 成绩不及格")  
    }  
}

程序返回:

成绩等级为B

这里程序根据变量的值而选择执行不同的分支代码,但需要注意的是,Go lang对于 { 和 } 的位置有严格的要求,它要求 else if (或 else ) 和两边的花括号,必须在同一行。即使在 { 和 } 之间只有一条语句,这两个花括号也是不能省略的。

选择结构之选择判断switch

switch关键字是一个条件语句,它计算表达式并将其与可能匹配的列表进行比较,并根据匹配执行代码块。它可以被理解为用一种普适的方式来写多个if else判断子句。

switch 语句用于基于不同条件执行不同动作,每一个 case 分支都是唯一的,从上直下逐一测试,直到匹配为止。 switch 语句执行的过程从上至下,直到找到匹配项,匹配项后面也不需要再加break。

说白了就是,每一个case都默认自动break,执行完了一个,switch逻辑也就结束了,不会顺序执行别的case,但是可以使用fallthrough强制执行后面的case代码:



package main  
  
import "fmt"  
  
func main() {  
    /* 定义局部变量 */  
    var grade string = "B"  
    var marks int = 40  
  
    switch marks {  
    case 90:  
        grade = "A"  
    case 80:  
        grade = "B"  
    case 50, 60, 70:  
        grade = "C" //case 后可以由多个数值  
    default:  
        grade = "D"  
    }  
  
    switch {  
    case grade == "A":  
        fmt.Printf("A\n")  
    case grade == "B", grade == "C":  
        fmt.Printf("B\n")  
    case grade == "D":  
        fmt.Printf("D\n")  
    case grade == "F":  
        fmt.Printf("F\n")  
    default:  
        fmt.Printf("low\n")  
    }  
    fmt.Printf("你的等级是 %s\n", grade)  
}  


程序返回:

D  
你的等级是 D

这里我们先通过switch对marks变量进行值判断,在case分支里赋值grade变量,随后又在switch逻辑中对grade做恒等判断,然后输出。

假设需要贯通后续的case,就添加fallthrough关键字:

package main  
  
import (  
    "fmt"  
)  
  
func main() {  
    switch x := 5; x {  
    default:  
        fmt.Println(x)  
    case 5:  
        x += 10  
        fmt.Println(x)  
        fallthrough  
    case 6:  
        x += 20  
        fmt.Println(x)  
  
    }  
  
}

这里首先进入case5逻辑,x经过运算后变为15,随后贯通进入下一个逻辑,x += 20 逻辑,变为35。

程序返回:

15  
35

需要注意的是,fallthrough应该是某个case的最后一行。如果它出现在中间的某个地方,编译器就会抛出错误。

循环结构之遍历for

for关键字可以用来重复执行某一段代码,在Python中,遍历方式有三种:for 、 while 和 do while 。但是Go lang只为我们提供了一种:for:

package main  
  
import (  
    "fmt"  
)  
  
func main() {  
    num := 1  
    for num < 3 {  
        fmt.Println(num)  
        num++  
    }  
  
}

程序返回:

1  
2

这里是但条件循环,如果满足条件,for代码块的逻辑会重置执行。

我们还可以为遍历添加额外的表达式逻辑,比如初始化控制变量,在整个循环生命周期内,只执行一次;设置循环控制条件,该表达式值为 true 时循环,值为 false 时结束循环;每次循环完都会执行此表达式,可以利用其让控制变量增量或减量:

package main  
  
import (  
    "fmt"  
)  
  
func main() {  
    for num := 0; num < 4; num++ {  
        fmt.Println(num)  
    }  
  
}

程序返回:

0  
1  
2  
3

需要注意的是,在for关键字中声明的变量,也只在for的代码块中有效,因为和Python不同,go lang有严格的块作用域限制。

在 Go lang中遍历一个可迭代的对象一般使用 for-range 语句实现,其中 range 后面可以接数组、切片、字符串等, range 会返回两个值,第一个是索引值,第二个是数据值:

package main  
  
import (  
    "fmt"  
)  
  
func main() {  
  
    str := "123456789"  
    for index, value := range str {  
        fmt.Printf("index %d, value %c\n", index, value)  
    }  
  
}

程序返回:

index 0, value 1  
index 1, value 2  
index 2, value 3  
index 3, value 4  
index 4, value 5  
index 5, value 6  
index 6, value 7  
index 7, value 8  
index 8, value 9

如果for关键字后面没有表单式,就是死循环:

package main  
  
import (  
    "fmt"  
)  
  
func main() {  
  
    num := 1  
  
    for {  
  
        fmt.Println(num)  
        num++  
  
        if num > 3 {  
            break  
        }  
  
    }  
  
}

程序返回:

1  
2  
3

是的,我们当然可以使用break关键字来中断循环。

但需要注意的是,break只能终端当前循环,不能终端外部循环:

package main  
  
import "fmt"  
  
func main() {  
    /* 定义局部变量 */  
    var i, j int  
    for i = 2; i < 10; i++ {  
        for j = 2; j <= (i / j); j++ {  
            if i%j == 0 {  
                break // 如果发现因子,则不是素数  
            }  
        }  
        if j > (i / j) {  
            fmt.Printf("%d  是素数\n", i)  
        }  
    }  
}

程序返回:

2  是素数  
3  是素数  
5  是素数  
7  是素数

和Python一样,Go lang也具备continue关键字,continue 语句用来跳出 for 循环中的当前循环:

package main  
  
import "fmt"  
  
func main() {  
  
    for num := 1; num <= 10; num++ {  
        if num%2 == 0 {  
            continue  
        }  
        fmt.Println(num)  
    }  
  
}

程序返回:

1  
3  
5  
7  
9

在 continue 语句后的所有的 for 循环语句都不会在本次循环中执行,执行完 continue 语句后将会继续执行一下次循环。这样我们就可以跳过偶数,直接打印出 10 以内的奇数。

goto 无条件跳转

使用goto关键字可以直接跳到下一步要执行的标签代码:

package main  
  
import "fmt"  
  
func main() {  
  
    for x := 0; x < 10; x++ {  
  
        for y := 0; y < 10; y++ {  
  
            if y == 2 {  
                // 跳转到标签  
                goto breakHere  
            }  
  
        }  
    }  
  
    // 手动返回, 避免执行进入标签  
    return  
  
    // 标签  
breakHere:  
    fmt.Println("done")  
}

程序返回:

done

需要注意的是,goto关键字与标签之间不能有变量声明,否则编译错误。

结语

和Python和Ruby相比,整体上,在流程结构控制环节,Go lang表现出了极大的克制,语法上删繁就简, 把动态语言那些桀骜不驯的语法糖压制成行文工整的诗,这样的好处是对初学者极为友好,没有太多规范需要记忆,大道至简,大巧不工。我们可以吐槽它没有while或者是do while,亦或者是lambda表达式等可以炫技的资本,但,那又如何呢?就像乔帮主在聚贤庄力战群雄,大杀四方,所倚重的不过是一套普普通通的太祖长拳,Go lang亦如此,务实胜过炫技,简单未必普通。

相关文章
|
16天前
|
人工智能 安全 算法
Go入门实战:并发模式的使用
本文详细探讨了Go语言的并发模式,包括Goroutine、Channel、Mutex和WaitGroup等核心概念。通过具体代码实例与详细解释,介绍了这些模式的原理及应用。同时分析了未来发展趋势与挑战,如更高效的并发控制、更好的并发安全及性能优化。Go语言凭借其优秀的并发性能,在现代编程中备受青睐。
80 33
|
1月前
|
存储 算法 数据可视化
【二叉树遍历入门:从中序遍历到层序与右视图】【LeetCode 热题100】94:二叉树的中序遍历、102:二叉树的层序遍历、199:二叉树的右视图(详细解析)(Go语言版)
本文详细解析了二叉树的三种经典遍历方式:中序遍历(94题)、层序遍历(102题)和右视图(199题)。通过递归与迭代实现中序遍历,深入理解深度优先搜索(DFS);借助队列完成层序遍历和右视图,掌握广度优先搜索(BFS)。文章对比DFS与BFS的思维方式,总结不同遍历的应用场景,为后续构造树结构奠定基础。
138 10
|
2月前
|
Go 开发者
go-carbon v2.6.0 重大版本更新,轻量级、语义化、对开发者友好的 golang 时间处理库
carbon 是一个轻量级、语义化、对开发者友好的 Golang 时间处理库,提供了对时间穿越、时间差值、时间极值、时间判断、星座、星座、农历、儒略日 / 简化儒略日、波斯历 / 伊朗历的支持
76 3
|
3月前
|
存储 Go
Go 语言入门指南:切片
Golang中的切片(Slice)是基于数组的动态序列,支持变长操作。它由指针、长度和容量三部分组成,底层引用一个连续的数组片段。切片提供灵活的增减元素功能,语法形式为`[]T`,其中T为元素类型。相比固定长度的数组,切片更常用,允许动态调整大小,并且多个切片可以共享同一底层数组。通过内置的`make`函数可创建指定长度和容量的切片。需要注意的是,切片不能直接比较,只能与`nil`比较,且空切片的长度为0。
Go 语言入门指南:切片
|
3月前
|
Go C语言
Go语言入门:分支结构
本文介绍了Go语言中的条件语句,包括`if...else`、`if...else if`和`switch`结构,并通过多个练习详细解释了它们的用法。`if...else`用于简单的条件判断;`if...else if`处理多条件分支;`switch`则适用于基于不同值的选择逻辑。特别地,文章还介绍了`fallthrough`关键字,用于优化重复代码。通过实例如判断年龄、奇偶数、公交乘车及成绩等级等,帮助读者更好地理解和应用这些结构。
58 15
|
6月前
|
JSON Go 开发者
go-carbon v2.5.0 发布,轻量级、语义化、对开发者友好的 golang 时间处理库
carbon 是一个轻量级、语义化、对开发者友好的 Golang 时间处理库,提供了对时间穿越、时间差值、时间极值、时间判断、星座、星座、农历、儒略日 / 简化儒略日、波斯历 / 伊朗历的支持。
157 4
|
3月前
|
运维 监控 算法
监控局域网其他电脑:Go 语言迪杰斯特拉算法的高效应用
在信息化时代,监控局域网成为网络管理与安全防护的关键需求。本文探讨了迪杰斯特拉(Dijkstra)算法在监控局域网中的应用,通过计算最短路径优化数据传输和故障检测。文中提供了使用Go语言实现的代码例程,展示了如何高效地进行网络监控,确保局域网的稳定运行和数据安全。迪杰斯特拉算法能减少传输延迟和带宽消耗,及时发现并处理网络故障,适用于复杂网络环境下的管理和维护。
|
3月前
|
编译器 Go
揭秘 Go 语言中空结构体的强大用法
Go 语言中的空结构体 `struct{}` 不包含任何字段,不占用内存空间。它在实际编程中有多种典型用法:1) 结合 map 实现集合(set)类型;2) 与 channel 搭配用于信号通知;3) 申请超大容量的 Slice 和 Array 以节省内存;4) 作为接口实现时明确表示不关注值。此外,需要注意的是,空结构体作为字段时可能会因内存对齐原因占用额外空间。建议将空结构体放在外层结构体的第一个字段以优化内存使用。
|
3月前
|
算法 安全 Go
公司局域网管理系统里的 Go 语言 Bloom Filter 算法,太值得深挖了
本文探讨了如何利用 Go 语言中的 Bloom Filter 算法提升公司局域网管理系统的性能。Bloom Filter 是一种高效的空间节省型数据结构,适用于快速判断元素是否存在于集合中。文中通过具体代码示例展示了如何在 Go 中实现 Bloom Filter,并应用于局域网的 IP 访问控制,显著提高系统响应速度和安全性。随着网络规模扩大和技术进步,持续优化算法和结合其他安全技术将是企业维持网络竞争力的关键。
82 2
公司局域网管理系统里的 Go 语言 Bloom Filter 算法,太值得深挖了
|
3月前
|
监控 Linux PHP
【02】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-2月12日优雅草简化Centos stream8安装zabbix7教程-本搭建教程非docker搭建教程-优雅草solution
【02】客户端服务端C语言-go语言-web端PHP语言整合内容发布-优雅草网络设备监控系统-2月12日优雅草简化Centos stream8安装zabbix7教程-本搭建教程非docker搭建教程-优雅草solution
114 20