Go初探

官方网站:https://golang.org/
标准库文档:https://golang.org/pkg/
在线编码学习:https://play.golang.org/
PS:请自行FQ

简介
安装
Hello World 实例
编译
基础语法
行分隔符
注释
标识符
关键字
数据类型
指针类型(Pointer)
数组(Array)
结构化类型(struct)
接口类型(interface)
Map 类型
Any类型
类型转换
类型增加方法
变量作用域
值类型和引用类型
常量
iota
运算符
运算符优先级
条件与循环语句
函数
返回多个值
参数
匿名函数与闭包
范围(Range)
递归
查询
接口查询
类型查询
错误处理
面向对象
成员的可访问性
封装
继承
多态
goroutine和channel
同步
消息传递
多个channel
参考

简介


(大家好,我是go吉祥物,地鼠君)

Go语言是谷歌推出的一种全新的编程语言,可以在不损失应用程序性能的情况下降低代码的复杂性。谷歌首席软件工程师罗布派克(Rob Pike)说:我们之所以开发Go,是因为过去10多年间软件开发的难度令人沮丧。

Go是从2007年末由Robert Griesemer, Rob Pike, Ken Thompson主持开发,后来还加入了Ian Lance Taylor, Russ Cox等人(真是豪门出身),并最终于2009年11月开源,在2012年早些时候发布了Go 1稳定版本。现在Go的开发已经是完全开放的,并且拥有一个活跃的社区。

Go语言专门针对多处理器系统应用程序的编程进行了优化,使用Go编译的程序可以媲美C或C++代码的速度,而且更加安全、支持并行进程。

Go的目标是希望提升现有编程语言对程序库等依赖性(dependency)的管理,这些软件元素会被应用程序反复调用。由于存在并行编程模式,因此这一语言也被设计用来解决多处理器的任务。

如果第一次接触Go语言,更像是对c语言的修补和延伸。

特性

  • 编译型语言,执行效率接近c/c++,可以编译成机器码,不依赖其他库
  • 自动垃圾回收
  • 更丰富的内置类型和自动类型推导(类似c++11的auto)
  • 函数可以返回多个值
  • 拥有更好的错误处理
  • 匿名函数和闭包
  • 支持类型和接口
  • 并发编程
  • 支持反射机制
  • 语言交互性强
  • 工程依赖自动推导
  • 打包和部署容易

安装

$ sudo apt-get install golang
$ sudo apt-get install gccgo
$ go version
go version go1.2.1 linux/amd64

Hello World 实例

  1. package main
  2. import"fmt"
  3. func main(){
  4. /* 这是我的第一个简单的程序 */
  5. fmt.Println("Hello, World!")
  6. }
  7. //执行
  8. $ go run hello.go
  9. Hello,World!

编译

Go是一个编译型的语言。目前有两种编译器,其中”Gccgo”采用GCC作为编译后端。另外还有 根据处理器架构命名的编译器:针对64位x86结构为”6g”,针对32位x86结构的为”8g”等等。 这些go专用的编译器编译很快,但是产生的目标代码效率比gccgo稍差一点。

$ 6g helloworld.go # 编译; 输出 helloworld.6
$ 6l helloworld.6 # 链接; 输出 6.out
$ 6.out
Hello, world!
$
$ gccgo helloworld.go
$ a.out
Hello, world!
$

基础语法

行分隔符

在 Go 程序中,一行代表一个语句结束。每个语句不需要像 C 家族中的其它语言一样以分号 ; 结尾,因为这些工作都将由 Go 编译器自动完成。
如果你打算将多个语句写在同一行,它们则必须使用 ; 人为区分,但在实际开发中我们并不鼓励这种做法。

  1. fmt.Println("Hello, World!")
  2. fmt.Println("w3cschool菜鸟教程:w3cschool.cc")
  3. fmt.Println("Hello, World!");fmt.Println("w3cschool菜鸟教程:w3cschool.cc")

注释

注释不会被编译,每一个包应该有相关注释。
单行注释是最常见的注释形式,你可以在任何地方使用以 // 开头的单行注释。多行注释也叫块注释,均已以 /* 开头,并以 */ 结尾。如:

  1. // 单行注释
  2. /*
  3. Author by w3cschool菜鸟教程
  4. 我是多行注释
  5. */

标识符

标识符用来命名变量、类型等程序实体。一个标识符实际上就是一个或是多个字母(A~Z和a~z)数字(0~9)、下划线_组成的序列,但是第一个字符必须是字母或下划线而不能是数字。

关键字

下面列举了 Go 代码中会使用到的 25 个关键字或保留字:
break default func interface select
case defer go map struct
chan else goto package switch
const fallthrough if range type
continue for import return var
除了以上介绍的这些关键字,Go 语言还有 36 个预定义标识符:
append bool byte cap close complex complex64 complex128 uint16
copy false float32 float64 imag int int8 int16 uint32
int32 int64 iota len make new nil panic uint64
print println real recover string true uint uint8 uintptr
程序一般由关键字、常量、变量、运算符、类型和函数组成。
程序中可能会使用到这些分隔符:括号 (),中括号 [] 和大括号 {}。
程序中可能会使用到这些标点符号:.、,、;、: 和 …。

数据类型

在 Go 编程语言中,数据类型用于声明函数和变量。
数据类型的出现是为了把数据分成所需内存大小不同的数据,编程的时候需要用大数据的时候才需要申请大内存,就可以充分利用内存。

布尔型
布尔型的值只可以是常量 true 或者 false。一个简单的例子:var b bool = true。

数字类型
整型 int 和浮点型 float,Go 语言支持整型和浮点型数字,并且原生支持复数,其中位的运算采用补码。
uint8
无符号 8 位整型 (0 到 255)
uint16
无符号 16 位整型 (0 到 65535)
uint32
无符号 32 位整型 (0 到 4294967295)
uint64
无符号 64 位整型 (0 到 18446744073709551615)
int8
有符号 8 位整型 (-128 到 127)
int16
有符号 16 位整型 (-32768 到 32767)
int32
有符号 32 位整型 (-2147483648 到 2147483647)
int64
有符号 64 位整型 (-9223372036854775808 到 9223372036854775807)
float32
IEEE-754 32位浮点型数
float64
IEEE-754 64位浮点型数
complex64
32 位实数和虚数
complex128
64 位实数和虚数
byte
类似 uint8
rune
类似 int32
uint
32 或 64 位
int
与 uint 一样大小
uintptr
无符号整型,用于存放一个指针

字符串类型:
字符串就是一串固定长度的字符连接起来的字符序列。Go的字符串是由单个字节连接起来的。Go语言的字符串的字节使用UTF-8编码标识Unicode文本。

指针类型(Pointer)

Go 语言中指针是很容易学习的,Go 语言中使用指针可以更简单的执行一些任务。
接下来让我们来一步步学习 Go 语言指针。
我们都知道,变量是一种使用方便的占位符,用于引用计算机内存地址。
Go 语言的取地址符是 &,放到一个变量前使用就会返回相应变量的内存地址。

一个指针变量可以指向任何一个值的内存地址它指向那个值的内存地址。
类似于变量和常量,在使用指针前你需要声明指针。
在指针类型前面加上 * 号(前缀)来获取指针所指向的内容。

  1. package main
  2. import"fmt"
  3. func main(){
  4. var a int=20/* 声明实际变量 */
  5. var ip *int/* 声明指针变量 */
  6. ip =&a /* 指针变量的存储地址 */
  7. fmt.Printf("a 变量的地址是: %x\n",&a )
  8. /* 指针变量的存储地址 */
  9. fmt.Printf("ip 变量的存储地址: %x\n", ip )
  10. /* 使用指针访问值 */
  11. fmt.Printf("*ip 变量的值: %d\n",*ip )
  12. }

当一个指针被定义后没有分配到任何变量时,它的值为 nil。
nil 指针也称为空指针。
nil在概念上和其它语言的null、None、nil、NULL一样,都指代零值或空值。
一个指针变量通常缩写为 ptr。

  1. if(ptr !=nil)/* ptr 不是空指针 */
  2. if(ptr ==nil)/* ptr 是空指针 */

如果一个指针变量存放的又是另一个指针变量的地址,则称这个指针变量为指向指针的指针变量。
当定义一个指向指针的指针变量时,第一个指针存放第二个指针的地址,第二个指针存放变量的地址:

  1. package main
  2. import"fmt"
  3. func main(){
  4. var a int
  5. var ptr *int
  6. var pptr **int
  7. a =3000
  8. /* 指针 ptr 地址 */
  9. ptr =&a
  10. /* 指向指针 ptr 地址 */
  11. pptr =&ptr
  12. /* 获取 pptr 的值 */
  13. fmt.Printf("变量 a = %d\n", a )
  14. fmt.Printf("指针变量 *ptr = %d\n",*ptr )
  15. fmt.Printf("指向指针的指针变量 **pptr = %d\n",**pptr)
  16. }

数组(Array)

  1. package main
  2. import"fmt"
  3. func main(){
  4. var n [10]int/* n 是一个长度为 10 的数组 */
  5. var i,j int
  6. /* 为数组 n 初始化元素 */
  7. for i =0; i <10; i++{
  8. n[i]= i +100/* 设置元素为 i + 100 */
  9. }
  10. /* 输出每个数组元素的值 */
  11. for j =0; j <10; j++{
  12. fmt.Printf("Element[%d] = %d\n", j, n[j])
  13. }
  14. }

结构化类型(struct)

Go 语言中数组可以存储同一类型的数据,但在结构体中我们可以为不同项定义不同的数据类型。
结构体是由一系列具有相同类型或不同类型的数据构成的数据集合。
Go语言的结构体(struct)和其它语言的类(class)有同等的地位。但Go语言放弃了包括继承在内的大量OOP特性,只保留了组合(compose)这个最基础的特性。
所有的Go语言的类型(指针类型除外)都是可以有自己的方法。在这个背景下,Go语言的结构体(struct)它只是很普通的复合类型

  1. package main
  2. import"fmt"
  3. type Booksstruct{
  4. title string
  5. author string
  6. subject string
  7. book_id int
  8. }
  9. func main(){
  10. varBook1Books/* 声明 Book1 为 Books 类型 */
  11. varBook2Books/* 声明 Book2 为 Books 类型 */
  12. /* book 1 描述 */
  13. Book1.title ="Go 语言"
  14. Book1.author ="www.runoob.com"
  15. Book1.subject ="Go 语言教程"
  16. Book1.book_id =6495407
  17. /* book 2 描述 */
  18. Book2.title ="Python 教程"
  19. Book2.author ="www.runoob.com"
  20. Book2.subject ="Python 语言教程"
  21. Book2.book_id =6495700
  22. /* 打印 Book1 信息 */
  23. printBook(Book1)
  24. /* 打印 Book2 信息 */
  25. printBook(Book2)
  26. }
  27. func printBook( book Books){
  28. fmt.Printf("Book title : %s\n", book.title);
  29. fmt.Printf("Book author : %s\n", book.author);
  30. fmt.Printf("Book subject : %s\n", book.subject);
  31. fmt.Printf("Book book_id : %d\n", book.book_id);
  32. }

构造函数?不需要。在Go语言中你只需要定义一个普通的函数,只是通常以NewXXX来命名,表示“构造函数”:
“`go
func NewRect(x, y, width, height float64) *Rect {
return &Rect{x, y, width, height}
}

  1. 这一切非常自然,没有任何突兀之处。
  2. ##切片(Slice)
  3. Go语言切片是对数组的抽象。
  4. Go数组的长度不可改变,在特定场景中这样的集合就不太适用,Go中提供了一种灵活,功能强悍的内置类型切片("动态数组"),与数组相比切片的长度是不固定的,可以追加元素,在追加时可能使切片的容量增大。
  5. 切片是可索引的,并且可以由 len()方法获取长度。
  6. 切片提供了计算容量的方法 cap()可以测量切片最长可以达到多少。
  7. 一个切片在未初始化之前默认为nil,长度为0
  8. 可以通过设置下限及上限来设置截取切片[lower-bound:upper-bound]
  9. ```go
  10. package main
  11. import "fmt"
  12. func main() {
  13. /* 创建切片 */
  14. numbers := []int{0,1,2,3,4,5,6,7,8}
  15. printSlice(numbers)
  16. /* 打印原始切片 */
  17. fmt.Println("numbers ==", numbers)
  18. /* 打印子切片从索引1(包含) 到索引4(不包含)*/
  19. fmt.Println("numbers[1:4] ==", numbers[1:4])
  20. /* 默认下限为 0*/
  21. fmt.Println("numbers[:3] ==", numbers[:3])
  22. /* 默认上限为 len(s)*/
  23. fmt.Println("numbers[4:] ==", numbers[4:])
  24. numbers1 := make([]int,0,5)
  25. printSlice(numbers1)
  26. /* 打印子切片从索引 0(包含) 到索引 2(不包含) */
  27. number2 := numbers[:2]
  28. printSlice(number2)
  29. /* 打印子切片从索引 2(包含) 到索引 5(不包含) */
  30. number3 := numbers[2:5]
  31. printSlice(number3)
  32. }
  33. func printSlice(x []int){
  34. fmt.Printf("len=%d cap=%d slice=%v\n",len(x),cap(x),x)
  35. }
  36. /*len=9 cap=9 slice=[0 1 2 3 4 5 6 7 8]
  37. numbers == [0 1 2 3 4 5 6 7 8]
  38. numbers[1:4] == [1 2 3]
  39. numbers[:3] == [0 1 2]
  40. numbers[4:] == [4 5 6 7 8]
  41. len=0 cap=5 slice=[]
  42. len=2 cap=9 slice=[0 1]
  43. len=3 cap=7 slice=[2 3 4]*/
  1. package main
  2. import"fmt"
  3. func main(){
  4. var numbers []int
  5. printSlice(numbers)
  6. /* 允许追加空切片 */
  7. numbers = append(numbers,0)
  8. printSlice(numbers)
  9. /* 向切片添加一个元素 */
  10. numbers = append(numbers,1)
  11. printSlice(numbers)
  12. /* 同时添加多个元素 */
  13. numbers = append(numbers,2,3,4)
  14. printSlice(numbers)
  15. /* 创建切片 numbers1 是之前切片的两倍容量*/
  16. numbers1 := make([]int, len(numbers),(cap(numbers))*2)
  17. /* 拷贝 numbers 的内容到 numbers1 */
  18. copy(numbers1,numbers)
  19. printSlice(numbers1)
  20. }
  21. func printSlice(x []int){
  22. fmt.Printf("len=%d cap=%d slice=%v\n",len(x),cap(x),x)
  23. }

接口类型(interface)

Go 语言提供了另外一种数据类型即接口,它把所有的具有共性的方法定义在一起,任何其他类型只要实现了这些方法就是实现了这个接口。

  1. package main
  2. import(
  3. "fmt"
  4. )
  5. type Phoneinterface{
  6. call()
  7. }
  8. type NokiaPhonestruct{
  9. }
  10. func (nokiaPhone NokiaPhone) call(){
  11. fmt.Println("I am Nokia, I can call you!")
  12. }
  13. type IPhonestruct{
  14. }
  15. func (iPhone IPhone) call(){
  16. fmt.Println("I am iPhone, I can call you!")
  17. }
  18. func main(){
  19. var phone Phone
  20. phone =new(NokiaPhone)
  21. phone.call()
  22. phone =new(IPhone)
  23. phone.call()
  24. }

Map 类型

Map 是一种无序的键值对的集合。Map 最重要的一点是通过 key 来快速检索数据,key 类似于索引,指向数据的值。
Map 是一种集合,所以我们可以像迭代数组和切片那样迭代它。不过,Map 是无序的,我们无法决定它的返回顺序,这是因为 Map 是使用 hash 表来实现的。

  1. package main
  2. import"fmt"
  3. func main(){
  4. /* 创建 map */
  5. countryCapitalMap := map[string]string{"France":"Paris","Italy":"Rome","Japan":"Tokyo","India":"New Delhi"}
  6. fmt.Println("原始 map")
  7. /* 打印 map */
  8. for country := range countryCapitalMap {
  9. fmt.Println("Capital of",country,"is",countryCapitalMap[country])
  10. }
  11. /* 删除元素 */
  12. delete(countryCapitalMap,"France");
  13. fmt.Println("Entry for France is deleted")
  14. fmt.Println("删除元素后 map")
  15. /* 打印 map */
  16. for country := range countryCapitalMap {
  17. fmt.Println("Capital of",country,"is",countryCapitalMap[country])
  18. }
  19. }

Any类型

由于Go语言中任何对象实例都满足空接口interface{},故此interface{}看起来像是可以指向任何对象的Any类型。如下:

  1. var v1 interface{}=1// 将int类型赋值给interface{}
  2. var v2 interface{}="abc"// 将string类型赋值给interface{}
  3. var v3 interface{}=&v2 // 将*interface{}类型赋值给interface{}
  4. var v4 interface{}=struct{ X int}{1}
  5. var v5 interface{}=&struct{ X int}{1}

当一个函数可以接受任意的对象实例时,我们会将其声明为interface{}。最典型的例子是标准库fmt中PrintXXX系列的函数。例如:

  1. func Printf(fmt string, args ...interface{})
  2. func Println(args ...interface{})

类型转换

类型转换用于将一种数据类型的变量转换为另外一种类型的变量。Go 语言类型转换基本格式如下:

type_name(expression)

type_name 为类型,expression 为表达式。

  1. package main
  2. import"fmt"
  3. func main(){
  4. var sum int=17
  5. var count int=5
  6. var mean float32
  7. mean = float32(sum)/float32(count)
  8. fmt.Printf("mean 的值为: %f\n",mean)
  9. }

类型增加方法

在Go语言中,你可以给任意类型(包括内置类型,但指针类型除外)增加方法,例如:

  1. type Integerint
  2. func (a Integer)Less(b Integer)bool{
  3. return a < b
  4. }

在Go语言中没有隐藏的this指针。这句话的含义是:

第一,方法施加的目标(也就是“对象”)显式传递,没有被隐藏起来。
第二,方法施加的目标(也就是“对象”)不需要非得是指针,也不用非得叫this。

在这个例子中,我们定义了一个新类型Integer,它和int没有本质不同,只是它为内置的int类型增加了个新方法:Less。如此,你就可以让整型看起来像个类那样用:
“`go
func main() {
var a Integer = 1
if a.Less(2) {
fmt.Println(a, “Less 2”)
}
}

  1. #变量
  2. Go语言变量名由字母、数字、下划线组成,其中首个字母不能为数字。
  3. 声明变量的一般形式是使用var关键字:
  4. ```go
  5. var identifier type
  6. //指定变量类型,声明后若不赋值,使用默认值。
  7. var v_name v_type
  8. v_name = value
  9. //根据值自行判定变量类型。
  10. var v_name = value
  11. v_name := value
  12. // 省略var, 注意 :=左侧的变量不应该是已经声明过的,否则会导致编译错误。
  13. var a int = 10
  14. var b = 10
  15. c : = 10
  16. //类型相同多个变量, 非全局变量
  17. var x, y int
  18. var ( // 这种因式分解关键字的写法一般用于声明全局变量
  19. a int
  20. b bool
  21. )
  22. var c, d int = 1, 2
  23. var e, f = 123, "hello"
  24. //这种不带声明格式的只能在函数体中出现
  25. //g, h := 123, "hello"
  26. func main(){
  27. g, h := 123, "hello"
  28. println(x, y, a, b, c, d, e, f, g, h)
  29. }

变量作用域

作用域为已声明标识符所表示的常量、类型、变量、函数或包在源代码中的作用范围。
Go 语言中变量可以在三个地方声明:

  • 函数内定义的变量称为局部变量
  • 函数外定义的变量称为全局变量
  • 函数定义中的变量称为形式参数

值类型和引用类型

多数Go语言中的类型,包括:

基本类型。如byte、int、bool、float32、float64、string等等。
复合类型。如数组(array)、结构体(struct)、指针(pointer)等。

都基于值语义。Go语言中类型的值语义表现得非常彻底。我们这么说是因为数组(array)。如果你学习过C语言,你会知道C语言中的数组(array)比较特别。通过函数传递一个数组的时候基于引用语义,但是在结构体中定义数组变量的时候是值语义(表现在结构体赋值的时候,该数组会被完整地拷贝一份新的副本)。

有4种引用类型:slice,map,channel,interface。

值类型的变量的值存储在栈(stack)中。
当使用等号 = 将一个变量的值赋值给另一个变量时,如:j = i,实际上是在内存中将 i 的值进行了拷贝
可以通过 &i 来获取变量 i 的内存地址,例如:0xf840000040(每次的地址都可能不一样)。
内存地址会根据机器的不同而有所不同,甚至相同的程序在不同的机器上执行后也会有不同的内存地址。因为每台机器可能有不同的存储器布局,并且位置分配也可能不同。
更复杂的数据通常会需要使用多个字,这些数据一般使用引用类型保存。

一个引用类型的变量 r1 存储的是 r1 的值所在的内存地址(数字),或内存地址中第一个字所在的位置
这个内存地址为称之为指针,这个指针实际上也被存在另外的某一个字中。
同一个引用类型的指针指向的多个字可以是在连续的内存地址中(内存布局是连续的),这也是计算效率最高的一种存储形式;也可以将这些字分散存放在内存中,每个字都指示了下一个字所在的内存地址。
当使用赋值语句 r2 = r1 时,只有引用(地址)被复制。
如果 r1 的值被改变了,那么这个值的所有引用都会指向被修改后的内容,在这个例子中,r2 也会受到影响。

常量

常量是一个简单值的标识符,在程序运行时,不会被修改的量。
常量中的数据类型只可以是布尔型、数字型(整数型、浮点型和复数)和字符串型。
常量的定义格式:

  1. package main
  2. import"fmt"
  3. func main(){
  4. const LENGTH int=10
  5. const WIDTH int=5
  6. var area int
  7. const a, b, c =1,false,"str"//多重赋值
  8. area = LENGTH * WIDTH
  9. fmt.Printf("面积为 : %d", area)
  10. println()
  11. println(a, b, c)
  12. }
  13. //常量还可以用作枚举
  14. const(
  15. Unknown=0
  16. Female=1
  17. Male=2
  18. )
  19. //常量可以用len(), cap(), unsafe.Sizeof()常量计算表达式的值。
  20. const(
  21. a ="abc"
  22. b = len(a)
  23. c =unsafe.Sizeof(a)
  24. )

iota

iota,特殊常量,可以认为是一个可以被编译器修改的常量。
在每一个const关键字出现时,被重置为0,然后再下一个const出现之前,每出现一次iota,其所代表的数字会自动增加1。

  1. package main
  2. import"fmt"
  3. func main(){
  4. const(
  5. a = iota //0
  6. b //1
  7. c //2
  8. d ="ha"//独立值,iota += 1
  9. e //"ha" iota += 1
  10. f =100//iota +=1
  11. g //100 iota +=1
  12. h = iota //7,恢复计数
  13. i //8
  14. )
  15. fmt.Println(a,b,c,d,e,f,g,h,i)
  16. #0 1 2 ha ha 100 100 7 8
  17. }

运算符

运算符用于在程序运行时执行数学或逻辑运算。
Go 语言内置的运算符有:(所有语言通用,略过)

  • 算术运算符
  • 关系运算符
  • 逻辑运算符
  • 位运算符
  • 赋值运算符
  • 其他运算符

其中
& 返回变量存储地址 &a; 将给出变量的实际地址。

  • 指针变量。 *a; 是一个指针变量

运算符优先级

有些运算符拥有较高的优先级,二元运算符的运算方向均是从左至右。下表列出了所有运算符以及它们的优先级,由上至下代表优先级由高到低:
^ !
* / % << >> & &^
+ - | ^
== != < <= >= >
<-
&&
||

条件与循环语句

大多数和C语言一样~

  1. package main
  2. import"fmt"
  3. func main(){
  4. var b int=15
  5. var a int
  6. numbers :=[6]int{1,2,3,5}
  7. /* for 循环 */
  8. for a :=0; a <10; a++{
  9. fmt.Printf("a 的值为: %d\n", a)
  10. /* 使用 break 语句跳出循环 */
  11. break;
  12. /* 跳过此次循环 */
  13. continue;
  14. }
  15. for a < b {
  16. a++
  17. fmt.Printf("a 的值为: %d\n", a)
  18. }
  19. for i,x:= range numbers {
  20. fmt.Printf("第 %d 位 x 的值 = %d\n", i,x)
  21. }
  22. /* 循环 */
  23. LOOP:for a <20{
  24. if a ==15{
  25. /* 跳过迭代 */
  26. a = a +1
  27. goto LOOP
  28. }
  29. fmt.Printf("a的值为 : %d\n", a)
  30. a++
  31. }
  32. fortrue{
  33. fmt.Printf("这是无限循环。\n");
  34. }
  35. }

函数

Go 语言最少有个 main() 函数。
你可以通过函数来划分不同功能,逻辑上每个函数执行的是指定的任务。
函数声明告诉了编译器函数的名称,返回类型,和参数。
Go 语言标准库提供了多种可动用的内置的函数。例如,len() 函数可以接受不同类型参数并返回该类型的长度。如果我们传入的是字符串则返回字符串的长度,如果传入的是数字,则返回数组中包含的函数个数。

  1. package main
  2. import"fmt"
  3. func main(){
  4. /* 定义局部变量 */
  5. var a int=100
  6. var b int=200
  7. var ret int
  8. /* 调用函数并返回最大值 */
  9. ret = max(a, b)
  10. fmt.Printf("最大值是 : %d\n", ret )
  11. }
  12. /* 函数返回两个数的最大值 */
  13. func max(num1, num2 int)int{
  14. /* 定义局部变量 */
  15. var result int
  16. if(num1 > num2){
  17. result = num1
  18. }else{
  19. result = num2
  20. }
  21. return result
  22. }

返回多个值

  1. package main
  2. import"fmt"
  3. func swap(x, y string)(string,string){
  4. return y, x
  5. }
  6. func main(){
  7. a, b := swap("Mahesh","Kumar")
  8. fmt.Println(a, b)
  9. }

参数

函数如果使用参数,该变量可称为函数的形参。
形参就像定义在函数体内的局部变量。
调用函数,可以通过两种方式来传递参数:
值传递:值传递是指在调用函数时将实际参数复制一份传递到函数中,这样在函数中如果对参数进行修改,将不会影响到实际参数。
引用传递:引用传递是指在调用函数时将实际参数的地址传递到函数中,那么在函数中对参数所进行的修改,将影响到实际参数。
默认情况下,Go 语言使用的是值传递,即在调用过程中不会影响到实际参数。

匿名函数与闭包

Go 语言支持匿名函数,可作为闭包(闭包就是能够读取其他函数内部变量的函数)。匿名函数是一个”内联”语句或表达式。匿名函数的优越性在于可以直接使用函数内的变量,不必申明。

  1. package main
  2. import"fmt"
  3. func getSequence() func()int{
  4. i:=0
  5. return func()int{
  6. i+=1
  7. return i
  8. }
  9. }
  10. func main(){
  11. /* nextNumber 为一个函数,函数 i 为 0 */
  12. nextNumber := getSequence()
  13. /* 调用 nextNumber 函数,i 变量自增 1 并返回 */
  14. fmt.Println(nextNumber())
  15. fmt.Println(nextNumber())
  16. fmt.Println(nextNumber())
  17. /* 创建新的函数 nextNumber1,并查看结果 */
  18. nextNumber1 := getSequence()
  19. fmt.Println(nextNumber1())
  20. fmt.Println(nextNumber1())
  21. }

范围(Range)

Go 语言中 range 关键字用于for循环中迭代数组(array)、切片(slice)、链表(channel)或集合(map)的元素。在数组和切片中它返回元素的索引值,在集合中返回 key-value 对的 key 值。

  1. package main
  2. import"fmt"
  3. func main(){
  4. //这是我们使用range去求一个slice的和。使用数组跟这个很类似
  5. nums :=[]int{2,3,4}
  6. sum :=0
  7. for _, num := range nums {
  8. sum += num
  9. }
  10. fmt.Println("sum:", sum)
  11. //在数组上使用range将传入index和值两个变量。上面那个例子我们不需要使用该元素的序号,所以我们使用空白符"_"省略了。有时侯我们确实需要知道它的索引。
  12. for i, num := range nums {
  13. if num ==3{
  14. fmt.Println("index:", i)
  15. }
  16. }
  17. //range也可以用在map的键值对上。
  18. kvs := map[string]string{"a":"apple","b":"banana"}
  19. for k, v := range kvs {
  20. fmt.Printf("%s -> %s\n", k, v)
  21. }
  22. //range也可以用来枚举Unicode字符串。第一个参数是字符的索引,第二个是字符(Unicode的值)本身。
  23. for i, c := range "go"{
  24. fmt.Println(i, c)
  25. }
  26. }
  27. /*
  28. sum: 9
  29. index: 1
  30. a -> apple
  31. b -> banana
  32. 0 103
  33. 1 111
  34. */

递归

递归,就是在运行的过程中调用自己。
Go 语言支持递归。但我们在使用递归时,开发者需要设置退出条件,否则递归将陷入无限循环中。
递归函数对于解决数学上的问题是非常有用的,就像计算阶乘,生成斐波那契数列等。

  1. package main
  2. import"fmt"
  3. func Factorial(x int)(result int){
  4. if x ==0{
  5. result =1;
  6. }else{
  7. result = x *Factorial(x -1);
  8. }
  9. return;
  10. }
  11. func main(){
  12. var i int=15
  13. fmt.Printf("%d 的阶乘是 %d\n", i,Factorial(i))
  14. }

查询

接口查询

  1. var file1 Writer=...
  2. if file5, ok := file1.(two.IStream); ok {
  3. ...
  4. }

这个if语句的含义是:file1接口指向的对象实例是否实现了two.IStream接口呢?如果实现了,则… 接口查询是否成功,要在运行期才能够确定。它不像接口赋值,编译器只需要通过静态类型检查即可判断赋值是否可行。
让语言内置接口查询,这是一件非常了不起的事情。
在Go语言中,你可以向接口询问,它指向的对象是否是某个类型

  1. var file1 Writer=...
  2. if file1, ok := file1.(*File); ok {
  3. ...
  4. }

这个if语句的含义是:file1接口指向的对象实例是否是 *File 类型呢?如果是的,则…
你可以认为查询接口所指向的对象是否是某个类型,只是接口查询的一个特例。接口是对一组类型的公共特性的抽象。

类型查询

在Go语言中,你还可以更加直接了当地询问接口指向的对象实例的类型。
就像现实生活中物种多得数不清一样,语言中的类型也多的数不清。所以类型查询并不经常被使用。它更多看起来是个补充,需要配合接口查询使用。

  1. type Stringerinterface{
  2. String()string
  3. }
  4. func Println(args ...interface{}){
  5. for _, arg := range args {
  6. switch v := v1.(type){
  7. caseint:// 现在v的类型是int
  8. casestring:// 现在v的类型是string
  9. default:
  10. if v, ok := arg.(Stringer); ok {// 现在v的类型是Stringer
  11. val := v.String()
  12. ...
  13. }else{
  14. ...
  15. }
  16. }
  17. }

利用反射(reflect)也可以进行类型查询,详细可参阅reflect.TypeOf方法相关文档。

错误处理

Go 语言通过内置的错误接口提供了非常简单的错误处理机制。
error类型是一个接口类型,这是它的定义:

  1. type error interface{
  2. Error()string
  3. }

我们可以在编码中通过实现 error 接口类型来生成错误信息。
函数通常在最后的返回值中返回错误信息。使用errors.New 可返回一个错误信息:

  1. func Sqrt(f float64)(float64, error){
  2. if f <0{
  3. return0, errors.New("math: square root of negative number")
  4. }
  5. // 实现
  6. }
  1. package main
  2. import(
  3. "fmt"
  4. )
  5. // 定义一个 DivideError 结构
  6. type DivideErrorstruct{
  7. dividee int
  8. divider int
  9. }
  10. // 实现 `error` 接口
  11. func (de *DivideError)Error()string{
  12. strFormat :=`
  13. Cannot proceed, the divider is zero.
  14. dividee: %d
  15. divider: 0
  16. `
  17. return fmt.Sprintf(strFormat, de.dividee)
  18. }
  19. // 定义 `int` 类型除法运算的函数
  20. func Divide(varDividee int, varDivider int)(result int, errorMsg string){
  21. if varDivider ==0{
  22. dData :=DivideError{
  23. dividee: varDividee,
  24. divider: varDivider,
  25. }
  26. errorMsg = dData.Error()
  27. return
  28. }else{
  29. return varDividee / varDivider,""
  30. }
  31. }
  32. func main(){
  33. // 正常情况
  34. if result, errorMsg :=Divide(100,10); errorMsg ==""{
  35. fmt.Println("100/10 = ", result)
  36. }
  37. // 当被除数为零的时候会返回错误信息
  38. if _, errorMsg :=Divide(100,0); errorMsg !=""{
  39. fmt.Println("errorMsg is: ", errorMsg)
  40. }
  41. }
  42. /*
  43. 100/10 = 10
  44. errorMsg is:
  45. Cannot proceed, the divider is zero.
  46. dividee: 100
  47. divider: 0
  48. */

面向对象

Go 语言的面向对象编程(OOP)非常简洁而优雅。说它简洁,是因为它没有了OOP中很多概念,比如:类,继承、虚函数、构造函数和析构函数、隐藏的this指针等等。说它优雅,是**它的面向对象(OOP)是语言类型系统(type system)中的天然的一部分。整个类型系统通过接口(interface)串联,浑然一体。
**

成员的可访问性

Go语言对关键字的增加非常吝啬。在Go语言中没有private、protected、public这样的关键字。要想某个符号可被其他包(package)访问,需要将该符号定义为大写字母开头。

  1. type Rectstruct{
  2. X, Y float64
  3. Width,Height float64
  4. }
  5. func (r *Rect) area() float64 {
  6. return r.Width* r.Height
  7. }

成员方法遵循同样的规则,Rect的area方法只能在该类型所在的包(package)内使用。
需要强调的一点是,Go语言中符号的可访问性是包(package)一级的,而不是类一级的。尽管area是Rect的内部方法,但是在同一个包中的其他类型可以访问到它。这样的可访问性控制很粗旷,很特别,但是非常实用。如果Go语言符号的可访问性是类一级的,少不了还要加上friend这样的关键字,以表示两个类是朋友关系,可以访问其中的私有成员。

封装

  1. package main
  2. import"fmt"
  3. type data struct{
  4. val int
  5. }
  6. //只有在你需要修改对象的时候,才必须用指针。它不是Go语言的约束,而是一种自然约束。
  7. //究其原因,是因为Go和C语言一样,类型都是基于值传递。要想修改变量的值,只能传递指针。
  8. func (p_data* data)set(num int){
  9. p_data.val = num
  10. }
  11. func (p_data* data)show(){
  12. fmt.Println(p_data.val)
  13. }
  14. func main(){
  15. p_data :=&data{4}
  16. p_data.set(5)
  17. p_data.show()
  18. }

继承

  1. package main
  2. import"fmt"
  3. type parent struct{
  4. val int
  5. }
  6. type child struct{
  7. parent
  8. num int
  9. }
  10. func main(){
  11. var c child
  12. c = child{parent{1},2}
  13. fmt.Println(c.num)
  14. fmt.Println(c.val)
  15. }

多态

  1. package main
  2. import"fmt"
  3. type act interface{
  4. write()
  5. }
  6. type xiaoming struct{
  7. }
  8. type xiaofang struct{
  9. }
  10. func (xm *xiaoming) write(){
  11. fmt.Println("xiaoming write")
  12. }
  13. func (xf *xiaofang) write(){
  14. fmt.Println("xiaofang write")
  15. }
  16. func main(){
  17. var w act;
  18. xm := xiaoming{}
  19. xf := xiaofang{}
  20. w =&xm
  21. w.write()
  22. w =&xf
  23. w.write()
  24. }

goroutine和channel

“网络,并发”是Go语言的两大feature。Go语言号称“互联网的C语言”,与使用传统的C语言相比,写一个Server所使用的代码更少,也更简单。写一个Server除了网络,另外就是并发,相对python等其它语言,Go对并发支持使得它有更好的性能。
Goroutine和channel是Go在“并发”方面两个核心feature。
Channel是goroutine之间进行通信的一种方式,它与Unix中的管道类似。

  1. ChannelType=("chan"|"chan""<-"|"<-""chan")ElementType.
  2. var ch chan int
  3. var ch1 chan<-int//ch1只能写
  4. var ch2 <-chan int//ch2只能读

channel是类型相关的,也就是一个channel只能传递一种类型。例如,上面的ch只能传递int。
创建channel时可以提供一个可选的整型参数,用于设置该channel的缓冲区大小。该值缺省为0,用来构建默认的“无缓冲channel”,也称为“同步channel”。
Channel作为goroutine间的一种通信机制,与操作系统的其它通信机制类似,一般有两个目的:同步,或者传递消息。

同步

  1. c := make(chan int)// Allocate a channel.
  2. // Start the sort in a goroutine; when it completes, signal on the channel.
  3. go func(){
  4. list.Sort()
  5. c <-1// Send a signal; value does not matter.
  6. }()
  7. doSomethingForAWhile()
  8. <-c // Wait for sort to finish; discard sent value.

上面的示例中,在子goroutine中进行排序操作,主goroutine可以做一些别的事情,然后等待子goroutine完成排序。
接收方会一直阻塞直到有数据到来。如果channel是无缓冲的,发送方会一直阻塞直到接收方将数据取出。如果channel带有缓冲区,发送方会一直阻塞直到数据被拷贝到缓冲区;如果缓冲区已满,则发送方只能在接收方取走数据后才能从阻塞状态恢复。

消息传递

经典的生产者-消费者模型

  1. func Producer(queue chan<-int){
  2. for i:=0; i <10; i++{
  3. queue <- i
  4. }
  5. }
  6. func Consumer( queue <-chan int){
  7. for i :=0; i <10; i++{
  8. v :=<- queue
  9. fmt.Println("receive:", v)
  10. }
  11. }
  12. func main(){
  13. queue := make(chan int,1)
  14. go Producer(queue)
  15. go Consumer(queue)
  16. time.Sleep(1e9)//让Producer与Consumer完成
  17. }

多个channel

在实际编程中,经常会遇到在一个goroutine中处理多个channel的情况。我们不可能阻塞在两个channel,这时就该select场了。与C语言中的select可以监控多个fd一样,go语言中select可以等待多个channel。

  1. c1 := make(chan string)
  2. c2 := make(chan string)
  3. go func(){
  4. time.Sleep(time.Second*1)
  5. c1 <-"one"
  6. }()
  7. go func(){
  8. time.Sleep(time.Second*2)
  9. c2 <-"two"
  10. }()
  11. for i :=0; i <2; i++{
  12. select{
  13. case msg1 :=<-c1:
  14. fmt.Println("received", msg1)
  15. case msg2 :=<-c2:
  16. fmt.Println("received", msg2)
  17. }
  18. }

参考

http://www.ituring.com.cn/article/1339
http://www.runoob.com/go/go-tutorial.html
http://blog.csdn.net/feixiaoxing/article/details/38167559

来自为知笔记(Wiz)

时间: 2024-10-13 14:26:10

Go初探的相关文章

进阶之初探nodeJS

一.前言 在"初探nodeJS"随笔中,我们对于node有了一个大致地了解,并在最后也通过一个示例,了解了如何快速地开启一个简单的服务器. 今儿,再次看了该篇随笔,发现该随笔理论知识稍多,适合初级入门node,固萌生一个想法--想在该篇随笔中,通过一步步编写一个稍大一点的node示例,让我们在整体上更加全面地了解node. so,该篇随笔是建立在"初探nodeJS"之上的,固取名为"进阶之初探nodeJS". 好了,侃了这多,那么我们即将实现一个

从273二手车的M站点初探js模块化编程

前言 这几天在看273M站点时被他们的页面交互方式所吸引,他们的首页是采用三次加载+分页的方式.也就说分为大分页和小分页两种交互.大分页就是通过分页按钮来操作,小分页是通过下拉(向下滑动)时异步加载数据. 273这个M站点是产品推荐我看的.第一眼看这个产品时我就再想他们这个三次加载和翻页按钮的方式,那么小分页的pageIndex是怎么计算的.所以就顺便看了下源码. 提到看源码时用到了Chrome浏览器的格式化工具(还是朋友推荐我的,不过这个格式化按钮的确不明显,不会的话自行百度). 三次加载和分

[转载]HDFS初探之旅

转载自 http://www.cnblogs.com/xia520pi/archive/2012/05/28/2520813.html , 感谢虾皮工作室这一系列精彩的文章. Hadoop集群(第8期)_HDFS初探之旅 1.HDFS简介 HDFS(Hadoop Distributed File System)是Hadoop项目的核心子项目,是分布式计算中数据存储管理的基础,是基于流数据模式访问和处理超大文件的需求而开发的,可以运行于廉价的商用服务器上.它所具有的高容错.高可靠性.高可扩展性.高

MongoDB初探系列之二:认识MongoDB提供的一些常用工具

在初探一中,我们已经可以顺利的将MongoDB在我们自己的机器上跑起来了.但是在其bin目录下面还有一些我们不熟知的工具.接下来,将介绍一下各个小工具的用途以及初探一中MongoDB在data文件夹下创建的文件的用途. 1.bin目录下面的各种小工具简介及使用方式 bsondump.exe 用于将导出的BSON文件格式转换为JSON格式mongo.exe mongoDB的客户端 mongod.exe 用于启动mongoDB的Server mongodump.exe 用于从mongodb数据库中导

Asynchronous Pluggable Protocols 初探

Asynchronous Pluggable Protocols,异步可插入协议,允许开发者创建可插协议处理器,MIME过滤器,以及命名空间处理器工作在微软IE4.0浏览器以及更高版本或者URL moniker中.这涉及到Urlmon.dll动态链接库所公开(输出)的可插协议诸多功能,本文不进行深入的原理讲解,只对它其中之一的应用进行解析,那就是如何将一个应用程序注册为URL协议. 应用场景: tencent协议: 当我们打开"tencent://message/?uin=要链接的QQ号 &qu

重新认识HTML,CSS,Javascript 之node-webkit 初探

今天我们来系统的.全面的 了解一下前端的一些技术,将有助于我们写出 更优秀的 产品 出来. 什么是HTML? HTML 是用来描述网页的一种语言. HTML 包含一些根节点,子节点,文本节点,属性节点,组成, 它通过一系列预定义标签来描述网页结构,如: <title>This is title</title> ,这个表明该网页的标题是 This is title. 什么是CSS? CSS 指层叠样式表 (Cascading Style Sheets),它描述浏览器显示如何显示htm

java进阶06 线程初探

线程,程序和进程是经常容易混淆的概念. 程序:就是有序严谨的指令集 进程:是一个程序及其数据在处理机上顺序执行时所发生的活动 线程:程序中不同的执行路径,就是程序中多种处理或者方法. 线程有两种方法实现 一:继承Thread 覆盖run方法 package Thread; public class Thread1 { public static void main(String[] args){ MyThread1 thread1=new MyThread1(); thread1.setName

数据加密解密初探

在一次网络通信或者是进程通信中,如果传输数据采用明文的方式,那么很容易被第三方"窃听"到,安全性难以保障. 而所谓加密是让数据从明文变成密文,传输过程中是密文,传送过去之后对方接收到的也是密文.--可以理解为密文就是乱码,看不出内在的任何意义,通常也都是逐位对应的. 在接收方接收到密文之后只有把它还原为原来的样子才可以理解对方说的具体是什么,此过程就叫做解密. 所谓系统的安全要实现的目标应该包括:机密性-confidentiality,完整性-integrity 和可用性-availa

Key/Value之王Memcached初探:三、Memcached解决Session的分布式存储场景的应用

一.高可用的Session服务器场景简介 1.1 应用服务器的无状态特性 应用层服务器(这里一般指Web服务器)处理网站应用的业务逻辑,应用的一个最显著的特点是:应用的无状态性. PS:提到无状态特性,不得不说下Http协议.我们常常听到说,Http是一个无状态协议,同一个会话的连续两个请求互相不了解,他们由最新实例化的环境进行解析,除了应用本身可能已经存储在全局对象中的所有信息外,该环境不保存与会话有关的任何信息.之所以我们在使用ASP.NET WebForm开发中会感觉不到Http的无状态特

Unity3D游戏开发初探

一.预备知识-对象的"生"与"死" (1)如何在游戏脚本程序中创建对象而不是一开始就创建好对象?->使用GameObject的静态方法:CreatePrimitive() 以上一篇的博文中的"指哪打哪"例子为基础,在AddForce脚本写入以下代码:   其中在CreateCube方法中,使用GameObject.CreatePrimitive方法来创建Cube类型的游戏对象实例,设置了它出现的坐标并为它增加刚体组件.这里可以看下AddCo