在前面的例子中,我们讲过 for 和 range为基本的数据结构提供了迭代的功能.我们也可以使用这个语法来遍历从通道中取得的值 Example: package main import "fmt" //这个例子也让我们看到,一个非空的通道也是可以关闭的 //但是通道中剩下的值仍然可以被接收到. func main() { jobs := make(chan string, 2) jobs <- "one" jobs <- "two"…
默认通道是 无缓冲 的,这意味着只有在对应的接收(<- chan)通道准备好接收时,才允许进行发送(chan <-).可缓存通道允许在没有对应接收方的情况下,缓存限定数量的值. 不支持缓冲: mk := make(chan string)通道不支持缓存,如果进行缓冲报如下错误.mk <- "chan" fatal error: all goroutines are asleep - deadlock! Example: package main import &quo…
我们可以使用通道来同步 Go 协程间的执行状态.这里是一个使用阻塞的接受方式来等待一个 Go 协程的运行结束. Example: package main import "fmt" import "time" //这是一个我们将要在 Go 协程中运行的函数.done 通道将被用于通知其他 Go 协程这个函数已经工作完毕. func worker(done chan bool){ fmt.Println("Working......") time.…
当使用通道作为函数的参数时,你可以指定这个通道是不是只用来发送或者接收值.这个特性提升了程序的类型安全性. Example: package main import "fmt" // <-chan 发送数据 // chan<- 接收数据 //ping 函数定义了一个只允许发送数据的通道.尝试使用这个通道来接收数据将会得到一个编译时错误. func ping(pings chan<- string, msg string){ pings<- msg } //pon…
Go 的通道选择器 让你可以同时等待多个通道操作.Go 协程和通道以及选择器的结合是 Go 的一个强大特性. Example: package main import "time" import "fmt" func main() { //在我们的例子中,我们将从两个通道中选择. c1 := make(chan string) c2 := make(chan string) //各个通道将在若干时间后接收一个值,这个用来模拟例如并行的 Go 协程中阻塞的 RPC 操…
range 迭代各种各样的数据结构.让我们来看看如何在我们已经学过的数据结构上使用 rang 吧. package main import "fmt" func main() { 这里我们使用 range 来统计一个 slice 的元素个数.数组也可以采用这种方法. nums := []int{2, 3, 4} sum := 0 for _, num := range nums { sum += num } fmt.Println("sum:", sum) rang…
第一篇文章(大约半年前写的):https://www.cnblogs.com/cgzl/p/11246324.html gRPC in ASP.NET Core 3.x -- Protocol Buffer(2)Go语言的例子(上) 接着上一篇文章. 写入到JSON 首先写一个func: 在这里,使用了jsonpb里面的Marshaler这个struct,这个struct上有一个方法叫做MarshalToString,它可以把proto.Message类型转化为JSON字符串. 与此同时,需要添…
通道 是连接多个 Go 协程的管道.你可以从一个 Go 协程将值发送到通道,然后在别的 Go 协程中接收. Example: package main import "fmt" func main(){ //使用 make(chan val-type) 创建一个新的通道.通道类型就是他们需要传递值的类型 mk := make(chan string) //使用 channel <- 语法 发送 一个新的值到通道中.这里我们在一个新的 Go 协程中发送 "ping&quo…
常规的通过通道发送和接收数据是阻塞的.然而,我们可以使用带一个 default 子句的 select 来实现非阻塞 的发送.接收,甚至是非阻塞的多路 select. Example: package main import "fmt" func main() { messages := make(chan string) signals := make(chan bool) //这里是一个非阻塞接收的例子.如果在 messages 中存在,然后 select 将这个值带入 <-m…
关闭 一个通道意味着不能再向这个通道发送值了.这个特性可以用来给这个通道的接收方传达工作已经完成的信息. Example: package main import "fmt" //在这个例子中,我们将使用一个 jobs 通道来传递 main() 中 Go协程任务执行的结束信息 //到一个工作 Go 协程中.当我们没有多余的任务给这个工作 Go 协程时 //我们将 close 这个 jobs 通道. func main() { jobs := make(chan int, 5) done…
SQL语言作为一种数据库管理的标准语言有着极为广泛的应用场景,菜鸟入门选用的数据库软件是轻量级的免费(这个极为重要)的MySQL,下载链接如下:http://www.mysql.com/downloads/ ,笔者的程序在MySQL可视化界面WorkBench下编写脚本文件并执行,这比在命令行模式下执行更加便捷且直观.安利一本MySQL菜鸟入门的书籍刘少杰和曾少军编著的<MySQL5.5——从零开始学>这本书(资源自寻,为避免侵权这里就不给出链接了,当然也欢迎直接买本纸质书支持作者). 例子:…
在前面的例子中,我们用互斥锁进行了明确的锁定来让共享的state 跨多个 Go 协程同步访问.另一个选择是使用内置的 Go协程和通道的的同步特性来达到同样的效果.这个基于通道的方法和 Go 通过通信以及 每个 Go 协程间通过通讯来共享内存,确保每块数据有单独的 Go 协程所有的思路是一致的. Example: package main import ( "fmt" "math/rand" "sync/atomic" "time&quo…
在前面的例子中,我们看到了如何使用原子操作来管理简单的计数器.对于更加复杂的情况,我们可以使用一个互斥锁来在 Go 协程间安全的访问数据. Example: package main import ( "fmt" "math/rand" "runtime" "sync" "sync/atomic" "time" ) func main() { //在我们的例子中,state 是一个 ma…
Go 中最主要的状态管理方式是通过通道间的沟通来完成的,我们在工作池的例子中碰到过,但是还是有一些其他的方法来管理状态的.这里我们将看看如何使用 sync/atomic包在多个 Go 协程中进行 原子计数 . Example: package main import "fmt" import "time" import "sync/atomic" import "runtime" func main() { //我们将使用一个…
速率限制(英) 是一个重要的控制服务资源利用和质量的途径.Go 通过 Go 协程.通道和打点器优美的支持了速率限制. Example: package main import "fmt" import "time" func main() { //首先我们将看一下基本的速率限制. //假设我们想限制我们接收请求的处理,我们将这些请求发送给一个相同的通道. requests := make(chan int, 5) for i:=1; i<=5; i++{ re…
在这个例子中,我们将看到如何使用 Go 协程和通道实现一个工作池 . Example: package main import "fmt" import "time" //这是我们将要在多个并发实例中支持的任务了.这些执行者将从 jobs 通道接收任务, //并且通过 results 发送对应的结果.我们将让每个任务间隔 1s 来模仿一个耗时的任务 func worker(id int, jobs <-chan int, results chan<- i…
定时器 是当你想要在未来某一刻执行一次时使用的 - 打点器 则是当你想要在固定的时间间隔重复执行准备的.这里是一个打点器的例子,它将定时的执行,直到我们将它停止. Example: package main import "fmt" import "time" func main() { //打点器和定时器的机制有点相似: //一个通道用来发送数据. //这里我们在这个通道上使用内置的 range 来迭代值每隔500ms 发送一次的值. ticker := time…
我们常常需要在后面一个时刻运行 Go 代码,或者在某段时间间隔内重复运行. Go 的内置 定时器 和 打点器 特性让这些很容易实现.我们将先学习定时器,然后再学习打点器. Example: package main import "fmt" import "time" func main() { //定时器表示在未来某一时刻的独立事件. //你告诉定时器需要等待的时间,然后它将提供一个用于通知的通道. //这里的定时器将等待 2 秒. timer1 := time.…
map 是 Go 内置关联数据类型(在一些其他的语言中称为哈希 或者字典 ) package main import "fmt" func main() { 要创建一个空 map,需要使用内建的 make:make(map[key-type]val-type). m := make(map[string]int) 使用典型的 make[key] = val 语法来设置键值对. m["k1"] = 7 m["k2"] = 13 使用例如 Print…
Go 内建多返回值 支持.这个特性在 Go 语言中经常被用到,例如用来同时返回一个函数的结果和错误信息. Example: package main import "fmt" //(int, int)标志函数有两个int返回值. func example_func(a int, b int) (int, int){ return a, b } func main(){ c, d := example_func(1, 3) //fmt.Printf用来格式化字符串. fmt.Printf…
Go 语言使用一个独立的·明确的返回值来传递错误信息的.这与使用异常的 Java 和 Ruby 以及在 C 语言中经常见到的超重的单返回值/错误值相比,Go 语言的处理方式能清楚的知道哪个函数返回了错误,并能像调用那些没有出错的函数一样调用. package main import "errors" import "fmt" 按照惯例,错误通常是最后一个返回值并且是 error 类型,一个内建的接口. func f1(arg int) (int, error) {…
超时 对于一个连接外部资源,或者其它一些需要花费执行时间的操作的程序而言是很重要的.得益于通道和 select,在 Go中实现超时操作是简洁而优雅的. Example: package main import "fmt" import "time" func main(){ c1 := make(chan string) //在我们的例子中,假如我们执行一个外部调用,并在 2 秒后通过通道 c1 返回它的执行结果. go func(){ time.Sleep(tim…
代码演示 package main import "fmt" func main() { queue := make(chan string, 2) queue <- "one" queue <- "two" close(queue) for elem := range queue { fmt.Println(elem) } } 代码运行结果 one two 代码解读 for和range为基本的数据结构提供了迭代功能,同样可以用于通道…
有时候我们想使用和集合的自然排序不同的方法对集合进行排序.例如,我们想按照字母的长度而不是首字母顺序对字符串排序.这里是一个 Go 自定义排序的例子. Example: package main import ( "fmt" "sort" ) //为了在 Go 中使用自定义函数进行排序, //我们需要一个对应的类型. //这里我们创建一个为内置 []string 类型的别名的byLength 类型, type byLength []string //我们在类型中实现…
Go 的 sort 包实现了内置和用户自定义数据类型的排序功能.我们首先关注内置数据类型的排序. Example: package main import ( "fmt" "sort" ) func main() { //排序方法是正对内置数据类型的:这里是一个字符串的例子. // 注意排序是原地更新的,所以他会改变给定的序列并且不返回一个新值 strarr := []string{"f", "d", "a"…
Example: package main import "fmt" func main() { fmt.Println("hello world") } Result: $ go run hello-world.go hello world 有时候我们想将我们的程序编译成二进制文件.我们可以通过 go build 命来达到目的. $ go build hello-world.go $ ls hello-world hello-world.go 然后我们可以直接运行…
Go 拥有各值类型,包括字符串,整形,浮点型,布尔型等.下面是一些基本的例子. Example: values.go package main import "fmt" func main() { 字符串可以通过 + 连接. fmt.Println("go" + "lang") 整数和浮点数 fmt.Println("1+1 =", 1+1) fmt.Println("7.0/3.0 =", 7.0/3.0…
在 Go 中,变量 被显式声明,并被编译器所用来检查函数调用时的类型正确性 package main import "fmt" func main() { var 声明 1 个或者多个变量. var a string = "initial" fmt.Println(a) 你可以申明一次性声明多个变量. var b, c int = 1, 2 fmt.Println(b, c) Go 将自动推断已经初始化的变量类型. var d = true fmt.Println(…
Go 支持字符.字符串.布尔和数值 常量 . package main import "fmt" import "math" const 用于声明一个常量. const s string = "constant" func main() { fmt.Println(s) const 语句可以出现在任何 var 语句可以出现的地方 const n = 500000000 常数表达式可以执行任意精度的运算 const d = 3e20 / n fmt…
for 是 Go 中唯一的循环结构.这里有 for 循环的三个基本使用方式. package main import "fmt" func main() { 最常用的方式,带单个循环条件. i := 1 for i <= 3 { fmt.Println(i) i = i + 1 } 经典的初始化/条件/后续形式 for 循环. for j := 7; j <= 9; j++ { fmt.Println(j) } 不带条件的 for 循环将一直执行,直到在循环体内使用了 br…