Go语言学习五 go的循环结构 chan range

原创 2018年04月17日 09:52:56

Go 语言循环语句


Go 语言循环语句

在不少实际问题中有许多具有规律性的重复操作,因此在程序中就需要重复执行某些语句。

以下为大多编程语言循环程序的流程图: 

Go 语言提供了以下几种类型循环处理语句:

循环类型描述
for 循环重复执行语句块
循环嵌套在 for 循环中嵌套一个或多个 for 循环

循环控制语句

循环控制语句可以控制循环体内语句的执行过程。

GO 语言支持以下几种循环控制语句:

控制语句描述
break 语句经常用于中断当前 for 循环或跳出 switch 语句
continue 语句跳过当前循环的剩余语句,然后继续进行下一轮循环。
goto 语句将控制转移到被标记的语句。

无限循环

如过循环中条件语句永远不为 false 则会进行无限循环,我们可以通过 for 循环语句中只设置一个条件表达式来执行无限循环:

package main

import "fmt"

func main() {
    for true  {
        fmt.Printf("这是无限循环。\n");
    }
}

Go 语言 for 循环

由 youj 创建, 最后一次修改 2016-07-22

Go 语言 for 循环

Go 语言循环语句Go 语言循环语句

for循环是一个循环控制结构,可以执行指定次数的循环。

语法

Go语言的For循环有3中形式,只有其中的一种使用分号。

和 C 语言的 for 一样:

for init; condition; post { }

和 C 的 while 一样:

for condition { }

和 C 的 for(;;) 一样:

for { }
  • init: 一般为赋值表达式,给控制变量赋初值;
  • condition: 关系表达式或逻辑表达式,循环控制条件;
  • post: 一般为赋值表达式,给控制变量增量或减量。

for语句执行过程如下:

  • ①先对表达式1赋初值;
  • ②判别赋值表达式 init 是否满足给定条件,若其值为真,满足循环条件,则执行循环体内语句,然后执行 post,进入第二次循环,再判别 condition;否则判断 condition 的值为假,不满足条件,就终止for循环,执行循环体外语句。

for 循环的 range 格式可以对 slice、map、数组、字符串等进行迭代循环。格式如下:

for key, value := range oldMap {
    newMap[key] = value
}

for语句语法流程如下图所示:

实例

package main

import "fmt"

func main() {

   var b int = 15
   var a int

   numbers := [6]int{1, 2, 3, 5} 

   /* for 循环 */
   for a := 0; a < 10; a++ {
      fmt.Printf("a 的值为: %d\n", a)
   }

   for a < b {
      a++
      fmt.Printf("a 的值为: %d\n", a)
      }

   for i,x:= range numbers {
      fmt.Printf("第 %d 位 x 的值 = %d\n", i,x)
   }   
}

以上实例运行输出结果为:

a 的值为: 0
a 的值为: 1
a 的值为: 2
a 的值为: 3
a 的值为: 4
a 的值为: 5
a 的值为: 6
a 的值为: 7
a 的值为: 8
a 的值为: 9
a 的值为: 1
a 的值为: 2
a 的值为: 3
a 的值为: 4
a 的值为: 5
a 的值为: 6
a 的值为: 7
a 的值为: 8
a 的值为: 9
a 的值为: 10
a 的值为: 11
a 的值为: 12
a 的值为: 13
a 的值为: 14
a 的值为: 15
第 0 位 x 的值 = 1
第 1 位 x 的值 = 2
第 2 位 x 的值 = 3
第 3 位 x 的值 = 5
第 4 位 x 的值 = 0

第 5 位 x 的值 = 0

Go 语言循环嵌套

由 youj 创建, 最后一次修改 2016-07-22

Go 语言循环嵌套

Go 语言循环语句Go 语言循环语句

Go 语言允许用户在循环内使用循环。接下来我们将为大家介绍嵌套循环的使用。

语法

以下为 Go 语言嵌套循环的格式:

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

实例

以下实例使用循环嵌套来输出 2 到 100 间的素数:

package main

import "fmt"

func main() {
   /* 定义局部变量 */
   var i, j int

   for i=2; i < 100; 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  是素数
11  是素数
13  是素数
17  是素数
19  是素数
23  是素数
29  是素数
31  是素数
37  是素数
41  是素数
43  是素数
47  是素数
53  是素数
59  是素数
61  是素数
67  是素数
71  是素数
73  是素数
79  是素数
83  是素数
89  是素数
97  是素数

Go 语言 break 语句

由 youj 创建, 最后一次修改 2016-07-22

Go 语言 break 语句

Go 语言循环语句 Go语言循环语句

Go 语言中 break 语句用于以下两方面:

  1. 用于循环语句中跳出循环,并开始执行循环之后的语句。
  2. break在switch(开关语句)中在执行一条case后跳出语句的作用。

语法

break 语法格式如下:

break;

break 语句流程图如下:

实例

package main

import "fmt"

func main() {
   /* 定义局部变量 */
   var a int = 10

   /* for 循环 */
   for a < 20 {
      fmt.Printf("a 的值为 : %d\n", a);
      a++;
      if a > 15 {
         /* 使用 break 语句跳出循环 */
         break;
      }
   }
}

以上实例执行结果为:

a 的值为 : 10
a 的值为 : 11
a 的值为 : 12
a 的值为 : 13
a 的值为 : 14
a 的值为 : 15

Go 语言 continue 语句

由 youj 创建, 最后一次修改 2016-07-22

Go 语言 continue 语句

Go 语言循环语句Go 语言循环语句

Go 语言的 continue 语句 有点像 break 语句。但是 continue 不是跳出循环,而是跳过当前循环执行下一次循环语句。

for 循环中,执行 continue 语句会触发for增量语句的执行。

语法

continue 语法格式如下:

continue;

continue 语句流程图如下:

实例

package main

import "fmt"

func main() {
   /* 定义局部变量 */
   var a int = 10

   /* for 循环 */
   for a < 20 {
      if a == 15 {
         /* 跳过此次循环 */
         a = a + 1;
         continue;
      }
      fmt.Printf("a 的值为 : %d\n", a);
      a++;     
   }  
}

以上实例执行结果为:

a 的值为 : 10
a 的值为 : 11
a 的值为 : 12
a 的值为 : 13
a 的值为 : 14
a 的值为 : 16
a 的值为 : 17
a 的值为 : 18
a 的值为 : 19

Go 语言 goto 语句

由 youj 创建, 最后一次修改 2016-07-22

Go 语言 goto 语句

Go 语言循环语句Go 语言循环语句

Go 语言的 goto 语句可以无条件地转移到过程中指定的行。

goto语句通常与条件语句配合使用。可用来实现条件转移, 构成循环,跳出循环体等功能。

但是,在结构化程序设计中一般不主张使用goto语句, 以免造成程序流程的混乱,使理解和调试程序都产生困难。

语法

goto 语法格式如下:

goto label;
..
.
label: statement;

break 语句流程图如下:

实例

package main

import "fmt"

func main() {
   /* 定义局部变量 */
   var a int = 10

   /* 循环 */
   LOOP: for a < 20 {
      if a == 15 {
         /* 跳过迭代 */
         a = a + 1
         goto LOOP
      }
      fmt.Printf("a的值为 : %d\n", a)
      a++     
   }  
}

以上实例执行结果为:

a的值为 : 10
a的值为 : 11
a的值为 : 12
a的值为 : 13
a的值为 : 14
a的值为 : 16
a的值为 : 17
a的值为 : 18
a的值为 : 19

go:chan分为阻塞和非阻塞


chan是一个FIFO队列,chan分成两种类型同步和异步
同步的chan完成发送者和接受者之间手递手传递元素的过程,必须要求对方的存在才能完成一次发送或接受
异步的chan发送和接受都是基于chan的缓存,但当缓存队列填满后,发送者就会进入发送队列, 当缓存队列为空时,接受者就会接入等待队列。

一句话总结:ch := make(chan int) 由于没有缓冲发送和接收需要同步,ch := make(chan int, 2) 有缓冲不要求发送和接收操作同步。

1、无缓冲时,发送阻塞直到数据被接收,接收阻塞直到读到数据


package main
import (
	"fmt"
	"time"
)
func main() {
	ch := make(chan int)
	go func() {
		var sum int = 6
		fmt.Println("first send")
		ch <- sum
		fmt.Println("second send start")
		fmt.Println("second send")
		ch <- sum
	}()
	time.Sleep(1e9)
	fmt.Println("first recv")
	fmt.Println(<-ch)
	fmt.Println("second recv")
	fmt.Println(<-ch)
}

first send

first recv

6

second recv

second send start

second send

6



2、有缓冲时,当缓冲满时发送阻塞,当缓冲空时接收阻塞


package main
import (
	"fmt"
	"time"
)
func main() {
	ch := make(chan int, 2)
	go func() {
		var sum int = 6
		fmt.Println("first send")
		ch <- sum
		fmt.Println("second send start")
		time.Sleep(1e9)
		fmt.Println("second send")
		ch <- sum
		fmt.Println("third send start")
		time.Sleep(1e9)
		fmt.Println("third send")
		ch <- sum
		time.Sleep(1e9)
		fmt.Println("forth send")
		ch <- sum
	}()
	time.Sleep(6e9)
	fmt.Println("first recv")
	fmt.Println(<-ch)
	fmt.Println("second recv")
	fmt.Println(<-ch)
	fmt.Println("third recv")
	fmt.Println(<-ch)
	fmt.Println("forth recv")
	fmt.Println(<-ch)
}

first send

second send start

second send

third send start

third send

first recv

6

second recv

6

third recv

6

forth recv

forth send

6


Go学习笔记之:range函数


range函数是个神奇而有趣的内置函数,你可以使用它来遍历数组,切片和字典。

当用于遍历数组和切片的时候,range函数返回索引和元素;

当用于遍历字典的时候,range函数返回字典的键和值。

package main

import "fmt"

func main() {

    // 这里我们使用range来计算一个切片的所有元素和
    // 这种方法对数组也适用
    nums := []int{2, 3, 4}
    sum := 0
    for _, num := range nums {
        sum += num
    }
    fmt.Println("sum:", sum)

    // range 用来遍历数组和切片的时候返回索引和元素值
    // 如果我们不要关心索引可以使用一个下划线(_)来忽略这个返回值
    // 当然我们有的时候也需要这个索引
    for i, num := range nums {
        if num == 3 {
            fmt.Println("index:", i)
        }
    }

    // 使用range来遍历字典的时候,返回键值对。
    kvs := map[string]string{"a": "apple", "b": "banana"}
    for k, v := range kvs {
        fmt.Printf("%s -> %s\n", k, v)
    }

    // range函数用来遍历字符串时,返回Unicode代码点。
    // 第一个返回值是每个字符的起始字节的索引,第二个是字符代码点,
    // 因为Go的字符串是由字节组成的,多个字节组成一个rune类型字符。
    for i, c := range "go" {
        fmt.Println(i, c)
    }
}

输出结果为

sum: 9
index: 1
a -> apple
b -> banana
0 103
1 111

go语言chan 和 routine活用

这里,我们以游戏中的一个情况为例。比如魔兽世界里的40人团队副本BOSS战,很多玩家同时攻击BOSS,BOSS的血量会进行平凡修改,我们要记录对BOSS的致命一击。常规 对BOSS血量的操作经行加锁,...
  • eclipser1987
  • eclipser1987
  • 2013-12-23 14:25:25
  • 11333

Go 生产者消费者模型

Go 生产者消费者模型标签(空格分隔): Go1.生产者消费者模型的实现func main() { in := make(chan int) out := make(chan in...
  • liu2012huan
  • liu2012huan
  • 2017-11-25 13:08:17
  • 155

GO 使用channel进行同步 (缓冲channel)

上一篇文章提到了普通的channel用来进行线程的同步。Go语言里面还有一种缓冲式的channel。 ch := make(chan int, 100) 需要执行类型和缓冲区的大小。 packa...
  • genispan
  • genispan
  • 2015-02-03 21:55:03
  • 3979

GO中chan的理解

chan的分类分为带缓存和不带缓存这2类,尤其需要关注带缓存的用法,防止掉坑里。 不带缓存 make(chan 数据类型) 带缓存 make(chan 数据类型,长度) 例如定义一个带缓存的cha...
  • u011768994
  • u011768994
  • 2016-12-09 10:47:18
  • 1901

Golang协程与通道整理

协程goroutine      不由OS调度,而是用户层自行释放CPU,从而在执行体之间切换。Go在底层进行协助实现      涉及系统调用的地方由Go标准库协助释放CPU      总之,不通...
  • max2005
  • max2005
  • 2013-07-11 17:00:01
  • 14098

Go语言 Channel <- 箭头操作符 详解

Channel是Go中的一个核心类型,你可以把它看成一个管道,通过它并发核心单元就可以发送或者接收数据进行通讯(communication)。 它的操作符是箭头  。 12 ch // ...
  • whatday
  • whatday
  • 2017-07-05 16:12:12
  • 3836

golang chan的运用

今天看gopl看到书上的例子看的不太懂,不是搞开发的,上了岁数理解能力也差,仔细的分析分析 package main import ( "bufio" "fmt" "log" "net" ) ...
  • yevvzi
  • yevvzi
  • 2016-09-20 21:46:29
  • 1814

Go语言并发与并行学习笔记(二)

Go语言的并发和并行 不知道你有没有注意到一个现象,还是这段代码,如果我跑在两个goroutines里面的话: var quit chan int = make(chan int) fu...
  • kjfcpua
  • kjfcpua
  • 2014-01-14 16:09:45
  • 10830

Go 通道(chan)关闭和后续读取操作

1、通道关闭时间: 2、有/无缓冲通道 3、range遍历通道
  • Tovids
  • Tovids
  • 2017-09-06 15:41:46
  • 813
收藏助手
不良信息举报
您举报文章:Go语言学习五 go的循环结构 chan range
举报原因:
原因补充:

(最多只允许输入30个字)