go语言基础概要

1 go语言特性

    自动垃圾回收,丰富的内置类型,函数多返回值,匿名函数与闭包,类型与非侵入式接口,并发编程(goroutine),反射机制。

2 基本语法

2.1 变量

2.1.1 变量声名方式

1. var v1 int or var v1 int = 3(声明并赋值)
2. var v2 = 3 (自动确定类型)
3. v3 := 3

2.1.2 其它

多重赋值:
i, j = j, i (交换值)
支持匿名变量(_)

2.2 常量

常量定义:
const Pi float64 = 3.14159
const(
    size int = 3
    eof = -1
)

const与itoa用法,一般可用于枚举型

2.3 数据类型

2.3.1 基本类型

1. 布尔类型:bool(只能接受true和false,不能接受其它类型赋值)
2. 整型:int8, byte(uint8),int16,uint等
3. 浮点类型:float32, float64
4. 复数类型:complex64, complex128(c := 3 + 3i)
5. 字符串:string (s1+s2, len(s), s[i], 两种遍历方式)
6. 字符:byte(utf-8字符),rune(unicode字符)
7. 错误类型:error

2.3.2 复合类型

1. 指针(类似c语言)
2. 数组(元素数量固定)array := [3]int{1,2,3}
3. 切片(动态数组)  
    slice := []int{1,2,3}
    slice := make([]int, 3) or make([]int, 3, 5) or 基于数组/切片创建
    cap(),len(),copy()和append()函数
4. 字典——map 
    map := map[keyType] valueType {key: value}
    map := make(map[keyType] valueType, 3)
    delete()和find()函数
5. 通道——chan
    gorontinue通信

2.4 流程控制

2.4.1 条件语句(if)

1. 条件语句不需要()
2. if之后,条件语句之前可以添加初始化语句,用;间隔
3. return语句不能包含在if...else...中
if v := 3 ; v >=3 {  //左花括号必须与if在同一行
    ...
    return r   // No
}

2.4.2 选择语句(switch)

1. 条件表达式不限定为常量或者整数
2. 不需要break来退出一个case
3. 只有在case中添加了fallthrouth关键字,才会继续执行下一个case
4. swicth后面可以没有条件表达式,相当于if...else...逻辑

2.4.3 循环语句(for)

1. 不需要()
2. 不支持使用,来实现多变量初始化,可以使用平行赋值语句
3. 支持break, continue

2.4.4 跳转语句(goto)

谨用

2.5 函数

2.5.1 函数定义

func name(arg1, arg2) (ret1, ret2){
    ...
}

2.5.2 特性

1. 支持多返回值
2. 不定参数
    func name(args ...int ) 参数个数不定,类型为int
    fun name(args ...interface{} ) 参数个数不定,类型不定
    调用方式:name(args...)
3. 匿名函数
    f := func(arg1, arg2) (ret1, ret2)
4. 闭包

2.6 错误处理

2.6.1 error接口

接口定义如下:
type error interface{
    Error() string
}
凡是实现了Error() string方法的类都属于error类型

2.6.1 defer

函数返回之前执行,一般用于释放资源等操作,当一个函数有多个defer时,调用顺序为先进后出,同时
支持匿名defer,定义如下:
defer func(){
    ...
}

2.6.1 panic()和recover()

panic函数

用于报告错误,即在一个函数中执行panic()时,正常的执行流程将终止,但是defer相应函数
依然执行,另外panic()会向上层执行panic()

recover函数

用于处理错误,一般用于defer中

3 面向对象

3.1 值与引用

go语言可以算都是基于值的传递,尤其对于数组,在c语言里是传地址,而在go中是传值(注意切片,map,
channel,interface的方式,实际也可以看成是传值)

3.2 结构体

3.2.1 定义与初始化

go中的结构体相当于其它语言中的class
定义 :
type Rect struct{
    x, y int
    width, height float
}
初始化:
r1 = new(Rect)
r2 = Rect{1,1,1,1}
r3 = Rect{width: 1, height:1}

3.2.2 继承

go采用组合的方法实现继承
例子如下:
type Base struct {
    Name string
}
func (base *Base) Foo() {...}
func (base *Base) Bar() {...}

type Foo struct {
    Base  or *Base   (注意区别)
    ...
}
func (foo *Foo) Bar() { 
    foo.Base.Bar()
    ...
}
上面首先定义基类Base,并且定义Foo()和Bar()两个方法;然后定义Foo(组合继承Base),而且
重写了Bar方法.foo.Foo()就会调用Base的Foo()方法
注意:命名冲突时,外层覆盖里层的

3.2.3 包可见性

首字母大写则表示对其它包可见

3.3 接口

3.3.1 非侵入式接口

一个类只要实现了某个接口的所有函数,就等于实现了这个接口;终于再也不用画类的继承树图了

3.3.2 接口赋值

1. 对象赋值给接口(两种方式:直接赋值与取地址赋值)
2. 接口赋值给接口
    两个接口如果方法相同,那么这两个接口实际上没有区别;另外当把接口A赋值给接口B时,B接口
    的方法应该是A接口方法的子集

3.3.2 接口查询

_, ok = interface1.(interface2)  interface1所指向的对象是否实现interface2接口的方法
特例 interface1.(Type) interface1所指向的对象是否属于Type类型
另外 interface1.(type) 获取interface1所指向的对象的类型(用在switch语句中)   

3.3.3 任意类型

var v1 interface{} = 1
var v1 interface{} = "string"

4 并发编程

4.1 并发模型

1. 多进程
2. 多纯种
3. 基于回调的异步非阻塞IO
4. 协程

4.2 goroutine

goroutine是Go语言的轻量级线程,由Go的runtime管理. 
使用方法: go functionName()

4.3 并发通信

共享数据和消息机制是两种常见的并发通信模型;Go语言采用消息机制的通信模型,称为channel.
"不要通过共享内存来通信,而应该通过通信来共享内存"

4.4 channel

4.4.1 基本语法

channel是类型相关的,定义方式如下:
var channelName chan Type
var ch chan float32 
var m map[string] chan bool
make(chan int)

单向channel:
var ch1 chan<- float32   写
var ch2 <- chan float32   读
var ch3 := chan<- float(ch)
var ch4 := <-chan float(ch)

存取channel(channel为空时,读取数据操作会阻塞;channel满时,存储数据操作会阻塞):
ch <- value
value := <-ch

关闭channel:
close(ch)

4.4.2 缓存

channel支持缓冲机制, ch = make(ch int, 1024)
for i := range ch {
    ...
}

4.4.3 超时机制

采用select实现超时机制:
timeout := make(chan bool, 1)
go func(){
    time.Sleep(le9)
    timeout <- true
}()
select {
    case <- ch :
    case <- timeout:
}

4.5 多核并行化

如若编译器版本不支持多核并行,则可以通过设置环境变量GOMAXPROCS的值:runtie.GOMAXPROCS(num)

4.5 同步

go语言也支持同步锁,包括sync.Mutex和sync.RWMutext

4.5 全局惟一性操作

sync.Once可以保证某个方法全局只执行一次,例如:
var once sync.Once
once.Do(func)  表示func函数全局只会执行一次

5 网络编程

5.1 socket编程

5.1.1 建立连接

TCP: conn, err = net.Dial("tcp","xxx.xxx.xxx.xxx:xxxx")
UDP: conn, err = net.Dial("udp","xxx.xxx.xxx.xxx:xxxx")
ICMP: conn, err = net.Dial("ip4:icmp","xxx.xxx.xxx.xxx:xxxx")

5.1.2 读写

conn.Write()   
conn.Read()

5.1.3 Dial()

Dial()函数其实是对DialTCP(),DialUDP(),DialIP()和DialUnix()的封装;所以在代码中可以直接
使用这些函数

5.2 http编程

5.2.1 Client

Client常用函数(http.Get(),http.Post()等)

func (c *Client) Get(url string) (r *Response, err error)
func (c *Client) Post(url string, bodyType string, body io.Reader) (r *Response,
 err error)
func (c *Client) PostForm(url string, data url.Values) (r *Response, err error)
func (c *Client) Head(url string) (r *Response, err error)
func (c *Client) Do(req *Request) (r *Response, err error)

高级封装

前面介绍的http.Get()实际上是http.DefaultClient.Get(),从而也说明Client是可以自定义的,
Client定义如下:
type Client type{
    //Transport用于确定http请求机制(是否长链接,ssl,连接数等)
    Transport RoundTripper
    //重定向之前调用该函数
    CheckRedirect fuc(req *Request, via []*Request) error
    //cookie信息
    Jar CookieJar
}
用户可以自定义Transport,CheckRedirect,Jar;从而根据需要实现自定义Client.

5.3 RPC

5.3.1 相关概念

标准库提供了net/rp包实现了RPC协议的相差细节. 在RPC服务端,可将一个对象注册为可访问的服务,之后
该对象的公开方法就能够以远程的方式提供访问。而且一个RPC服务端可以注册多个不同类型的对象,但是
不允许注册同一类型的多个对象.

对象中的方法需要满足以下要求才可以被远程访问:
1. 对象方法必需可供外部调用(首字母大写)
2. 必须有两个参数,第二个必须是指针
3. 方法必须返回error类型的值
如 func (t *T) MethodName(arg T1, reply *T2) error 

5.3.2 例子

服务端

obj := new(Obj)
rpc.Register(obj)
rpc.HandleHTTP()
l, e := net.Listen("tcp", ":3333")
if e! = nil {
    log.Fatal("listen error: ", e)
}
go http.Serve(l, nil)

客户端

client, err = rpc.DialHTTP("tcp", address+":3333")
同步:
err = client.Call()
异步:
asyCall := client.Go()
replyCall := <-asyCall.Done

5.4 JSON解析

5.4.1 编码

json.Marshal()
func Marshal(v interface{}) ([]byte, error)

5.4.2 解码

json.Unmarshal()
func Unmarshal(data []bite, v interface{}) error
注意:已知数据结构的解码与未知数据结构的解码

5.4.3 JSON流式读写

func NewDecoder(r io.Reader) *Decoder
func NewEncoder(w io.Writer) *Encoder

时间: 2016/01/28

你可能感兴趣的:(go)