Golang 学习日志

安装

官网下载地址:https://golang.org/dl/ ,根据系统平台下载对应资源包,安装或解压到对应目录,然后配置环境变量

GOROOT:go安装(非默认安装目录时需要配置)
PATH: go安装目录/bin(必需)
GOPATH:go项目目录(命令安装依赖时需要配置)

GOPATH目录约定有三个子目录:bin编译后生成的可执行文件,pkg编译后生成的文件,src存放源代码


认知

命名
  1. 驼峰式命名,不要使用下划线
  2. 根据首字母的大小写来确定可以访问的权限,无论是方法名、常量、变量名还是结构体的名称,如果首字母大写,则可以被其他的包访问(公有);如果首字母小写,则只能在本包中使用(私有)
变量
  1. var声明变量,函数内可用 := 简化声明和初始化
  2. _ 忽略返回值
  3. const 用于定义常量
包导入
import (
    "test/pkg1"       //默认导入方式,调用包内函数:pkg1.func1()
    p2 "test/pkg2"    //别名方式,调用包内函数:p2.func2()
    . "test/pkg3"     //省略前缀,调用包内函数:func3()
    _ "test/pkg4"     //仅执行包内init()函数,无法调用包内其他函数
)

常使用命令行 go get xxx 可以从指定源上面下载或者更新指定的代码和依赖,并对他们进行编译和安装

函数
  1. main() 一个包内只能有一个main函数
  2. init() 一个包内可以存在多个init函数,导入时都会被调用,在main函数之前被执行
  3. 常见写法
package main

import (
    "errors"
    "fmt"
)

func f1(num1 int, num2 int) int {
    return num1 + num2
}

func f2(num1 int, num2 int) (rs int) {
    rs = num1 + num2
    return
}

func f3(num1 int, num2 int) (rs int, err error) {
    if num2 == 0 {
        err = errors.New("num2 is empty.")
        return
    }
    rs = int(num1 / num2)
    return rs, nil
}

func main() {
    num1 := 5
    num2 := 3
    if rs, err := f3(num1, num2); err != nil {
        fmt.Println(err)
    } else {
        fmt.Printf("%d / %d = %d", num1, num2, rs)
    }
}
  1. defer代码块,在函数内部使用,会在函数结束或返回时被调用(先进后出)
func main() {
    fmt.Println(0)
    for i := 1; i <= 3; i++ {
        defer fmt.Println(i)
    }
    fmt.Println(4)
    // 0 4 3 2 1 
}
类型
  1. string 字符串类型,相当于[]byte
  2. interface{} 空接口类型,可以放入其他所有类型,一般用来定义未知类型的变量

类型转换

Golang是静态类型的编程语言,所有数据的类型在编译期确定。 Golang中即使是底层存的是同一个类型,声明的类型不一样,也要强制转换才能互用。

  1. Go语言类型转换基本格式如:type_name(expression)
    func main() {
        var a int = 1
        fmt.Printf("%T", a)             // int
        fmt.Printf("%T", float64(a))    // float64
    }
    
    如果强制转换一些无法转换的类型,将会报错

控制语句

  1. if - else
    // demo1
    if a == 1 {
        fmt.Println(1)
    } else if a == 2 {
        fmt.Println(2)
    } else {
        fmt.Println(3)
    }
    
    // demo2
    if b := a; b == 1 {
        fmt.Println(1)
    } else {
        fmt.Println(2)
    }
    
  2. switch - case
    var a int = 1
    
    // demo1
    switch a {
    case 1:
        fmt.Println(1)
    case 2:
        fmt.Println(2)
        fallthrough //连接执行下一个case
    case 3, 4, 5:
        fmt.Println(345)
    default:
        fmt.Println(0)
    }
    
    // demo2
    switch b := a; b {
    case 1:
        fmt.Println(1)
    default:
        fmt.Println(0)
    }
    
    // demo3
    switch {
    case a == 1:
        fmt.Println(1)
    case a == 2, a == 3:
        fmt.Println(23)
    default:
        fmt.Println(0)
    }
    
    // demo4 接口类型判断
    t := func() interface{} {
        return 1
    }()
    switch t.(type) {
    case int:
        fmt.Println("int")
    case string:
        fmt.Println("string")
    }
    
  3. for
    //基于计数器的迭代
    for i := 0; i < 10; i++ {
        fmt.Println(i)
    }
    
    //基于条件判断的迭代
    i := 10
    for i > 0 {
        fmt.Println(i)
        i--
    }
    
    //无限循环
    for {
        fmt.Println(1)
    }
    for true {
        fmt.Println(1)
    }
    
    //for-range迭代器
    //map,slice,array,channel
    for key, value := range array1 {
        fmt.Printf("%d => %v \n", key, value)
    }
    
  4. select - case
    可以用来处理channel操作
    select {
    case <-ch1:
        fmt.Println(1)
    case <-ch2:
        fmt.Println(2)
    case <-time.After(time.Minute):
        fmt.Println(9)
    default:
        fmt.Println(0)
    }
    

数组 Array

数组是类型相同的元素的集合

    var arr1 [5]int
    arr2 := [5]int{1, 2, 3}
    arr3 := [...]int{1, 2, 3}
    arr4 := [...]int{3: 3, 5: 5}
    arr5 := [4][2]int{{1, 2}, {3, 4}}

    fmt.Println(arr1, "\n", arr2, "\n", arr3, "\n", arr4, "\n", arr5)
    // [0 0 0 0 0]
    // [1 2 3 0 0]
    // [1 2 3]
    // [0 0 0 3 0 5]
    // [[1 2] [3 4] [0 0] [0 0]]

切片 Slice

切片是一个 长度可变的数组,是对数组一个连续片段的引用,是一个引用类型

//1.基于现有数组创建切片
var arr1 [10]int = [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9, 0}
//array[start:end] 不包含end,其中start和end在为第一或最后一个时可以省略
slice1 := arr1[3:7]
slice2 := arr1[:4]

//2,直接创建切片
//make([]T, length, capacity)
slice3 := make([]int, 2, 2)
slice4 := []int{1, 2, 3, 4}

//切片可以通过append,copy函数来进行追加和复制
slice4 = append(slice4, 5)
slice4 = append(slice4, 6)
slice5 := make([]int, len(slice4), cap(slice4)*2)
copy(slice5, slice4)
//在追加时,当切片的len超过cap时,切片的内存会重新分配,返回的新的切片不再指向原数组,
//新切片的cap会变为原来的2倍

集合map

Map是一种无序的键值对的集合
Map通过 key 来快速检索数据,key 类似于索引, 指向数据的值

//声明定义
var map1 map[string]int //nil
//map1 = make(map[string]int)
map1 = map[string]int{"a": 1, "b": 2}

//添加
map1["c"] = 3
//删除
delete(map1, "b")

//是否存在
if _, ok := map1["c"]; ok {
    fmt.Println("C is exist")
} else {
    fmt.Println("C is not exist")
}

自定义结构体 Struct

结构体由一系列属性组成,每个属性都有自己的类型和值

package main

import (
    "fmt"
)

type A struct {
    b int
    c int
    d int
}

func main() {
    var a1 A
    a1.b = 2
    fmt.Println(a1)
    // {2 0 0} 值类型

    a2 := new(A)
    a2.b = 2
    fmt.Println(a2)
    // &{2 0 0} 引用类型

    a3 := A{b: 3, c: 4}
    fmt.Println(a3)
    // {3 4 0} 值类型

    a4 := &A{3, 4, 5} //此写法需要全部参数赋值
    fmt.Println(a4)
    // &{3 4 5} 引用类型
}
  1. 初始化后, 字段默认为该类型的空值
  2. 直接定义A类型或使用A{}来完成初始化返回的是值类型,
    使用new()&A{}初始化返回的是引用类型, 两者不一样
struct 嵌套与函数扩展
package main

import (
    "fmt"
)

type A struct {
    a1 int
    a2 int
}

type B struct {
    A  // B嵌套A,B将可以使用A的参数和方法
    b1 int
    b2 int
}

func (a *A) Af() {
    fmt.Println("a func")
}

func (b *B) Bf() {
    fmt.Println("b func")
}

func main() {
    b := new(B)
    b.A.a1 = 1
    b.a2 = 2
    b.b1 = 3
    fmt.Println(b) // &{{1 2} 3 0}

    b.Af() //a func
    b.Bf() //b func
}

如果A、 B两个struct不在同一个包中, B将无法读写A中的私有字段和方法


接口 Interface

接口是一些方法的集合, 只要实现了接口对应的方法, 就等于实现了此接口,
因此golang中所有的类型都实现了空接口interface{},在一些函数中如果传入参数类型不固定,都可以使用interface{}代替传入,但在具体使用该参数时仍需要将类型转换回对应类型。

package main

import (
    "fmt"
)

type Ai interface {
    a()
    b()
}
type Ci interface {
    c()
}

type AC interface { // 接口嵌套
    Ai
    Ci
}

type T1 struct{}
type T2 struct{}

func (t *T1) a() {
    fmt.Println("a1")
}

func (t *T1) b() {
    fmt.Println("b1")
}

func (t *T1) c() {
    fmt.Println("c1")
}

func (t *T2) a() {
    fmt.Println("a2")
}

func (t *T2) c() {
    fmt.Println("c2")
}
func main() {

    //定义接口类型变量
    var test1 Ai
    test1 = new(T1) // T1实现了a()和b(),所以可以初始化T1类型赋值给Ai接口类型
    test1.a()       // 输出 a1
    //  var test2 Ai
    //  test2 = new(T2) // 无法编译通过,T2没有实现Ai接口的b(),所以无法初始化T1类型赋值给Ai接口类型

    //定义接口类型变量
    var test3, test4 Ci
    test3 = new(T1) // T1实现了c(),所以可以初始化T1类型赋值给Ci接口类型
    test3.c()       // 输出 c1
    test4 = new(T2) // T2实现了c(),所以可以初始化T2类型赋值给Ci接口类型
    test4.c()       // 输出 c2

    //定义接口类型变量
    var test5 AC
    test5 = new(T1) // T1实现了a()、b()、c(),所以可以初始化T1类型赋值给AC接口类型
    test5.a()       // a1
    test5.b()       // b1
    test5.c()       // c1

    //  var test6 AC
    //  test6 = new(T2) // 无法编译通过,T2没有实现AC接口的b(),所以无法初始化T2类型赋值给AC接口类型
}
接口类型转换
  • 断言x.(T) , x是接口类型,如str, ok := value.(string)
  • Type switch,详情见上述 控制语句 > switch-case > demo4

协程 Goroutines

协程是在一个线程中模拟多个协程并发执行
没有优先级、 非抢占式调度: 任务不能主动抢占时间片
每个协程都有自己的堆栈和局部变量
golang调度学习: https://www.jianshu.com/p/9db2dcb1ccb7
给调用函数使用 go 关键字即可将函数放到协程中执行

package main

import (
    "fmt"
)

func f1() {
    fmt.Println(1)
}
func f2() {
    fmt.Println(2)
}
func main() {
    go f1()
    f2()
}

反复执行上面代码,我们会发现很大概率上只会输出2,极小概率会同时输出1、2,因为在使用go激活 f1 后,主线程直接就往下执行 f2 ,最后主线程退出,其中 f1 还没来的及执行打印就已经结束了。

如何固定让主线程的 f2 在 f1 执行只会在执行呢?

  1. sync.WaitGroup
    package main
    
    import (
        "fmt"
        "sync"
    )
    
    func f1() {
        fmt.Println(1)
        wg.Done() // 完成一个计数器,计数 -1
    }
    func f2() {
        fmt.Println(2)
    }
    
    var wg sync.WaitGroup
    
    func main() {
        wg.Add(1) // 添加一个等待计数器,计数 +1
        go f1()
        wg.Wait() // 主线程堵塞等待,在计数器为0时才会向下执行
        f2()
    }
    
  2. 信道Channel
    可以通过它们发送类型化的数据在协程之间通信, 可以避开一些内存共享导致的坑通过通道传递的数据同一时间只有一个协程可以访问。
    package main
    
    import (
        "fmt"
    )
    
    func f1() {
        fmt.Println(1)
        ch <- 1 //将数据传入ch
    }
    func f2() {
        fmt.Println(2)
    }
    
    var ch chan int //声明信道和信道数据的类型
    
    func main() {
        ch = make(chan int) //初始化,分配内存
        go f1()
        <-ch //取出ch中的数据,若ch中没有数据,则ch会堵塞,直到ch中有数据传入
        f2()
    }
    
    在默认无缓冲通道情况下,如果通道中没有数据, 那 <-ch 数据取出就阻塞,
    如果通道已有数据, 那 ch<-value 数据存入就阻塞。有缓冲通道是指在未填满指定数量之前不会堵塞, 数量满了之后就会像无缓冲一样堵塞, 直到通道数据被取出
    ch := make(chan ch_type)
    // 无缓冲信道,ch_type可以为任意类型
    
    ch := make(chan ch_type, buf_num)
    //有缓冲信道,buf_num为缓冲数量
    
    
    ch <- 123
    // 将数据传入信道
    // 无缓冲时,信道将会堵塞,直到ch取出数据
    // 有缓冲时,信道将无限制传入,直到传入数量大于缓冲数量,才会堵塞,直到ch取出数据才会继续传入
    
    data := <- ch
    // 将数据从信道中取出,并赋给data变量
    // 信道将会堵塞,直到ch有数据传入
    

Http web服务

使用Golang的标准库 net/http可以搭建一个Go Web服务器

package main

import (
    "fmt"
    "log"
    "net/http"
)

func hello(w http.ResponseWriter, req *http.Request) {
    fmt.Fprintln(w, "hello world")
}

func main() {
    http.HandleFunc("/", hello)
    err := http.ListenAndServe("127.0.0.1:80", nil)
    if err != nil {
        log.Fatal("ListenAndServe:", err.Error())
    }
}

Request: 用户请求的信息,用来解析用户的请求信息,包括post、get、file等信息
Response: 服务器需要反馈给客户端的信息
Handler: 处理请求和生成返回信息的处理逻辑

在编写项目代码时,有时会觉得每个函数都要写(w http.ResponseWriter, req *http.Request),就会觉得很麻烦,所以做了以下测试

package main

import (
    "fmt"
    "log"
    "net/http"
)

type CTX struct {
    ResponseWriter http.ResponseWriter
    Request        *http.Request
}

var Ctx *CTX

func Handle(url string, f func()) {
    http.HandleFunc(url, func(w http.ResponseWriter, r *http.Request) {
        Ctx = &CTX{
            ResponseWriter: w,
            Request:        r,
        }
        f()
    })
}

func main() {
    Handle("/", home)
    Handle("/hello", hello)
    Handle("/ping", new(Ping).ping)
    err := http.ListenAndServe("127.0.0.1:80", nil)
    if err != nil {
        log.Fatal("ListenAndServe:", err.Error())
    }
}

////////////////////////////////////////////

func home() {
    fmt.Fprintln(Ctx.ResponseWriter, "home")
}

func hello() {
    fmt.Fprintln(Ctx.ResponseWriter, "hello world")
}

type Ping struct{}

func (p *Ping) ping() {
    fmt.Fprintln(Ctx.ResponseWriter, "pong")
}

错误类型

  1. 实现error接口
    type error interface{
        Error() string
    }
    
    package main
    
    import (
        "fmt"
    )
    
    type Myerror struct {
        msg string
    }
    
    func (e *Myerror) Error() string {
        return e.msg
    }
    
    func main() {
        var err error
        err = &Myerror{"test error"}
        fmt.Println(err) //test error
    }
    
  2. 使用errors包函数
    package main
    
    import (
        "errors"
        "fmt"
    )
    
    func main() {
        var err error
        err = errors.New("test error")
        fmt.Println(err) //test error
    }
    
  3. 使用fmt包函数
    package main
    
    import (
        "fmt"
    )
    
    func main() {
        var err error
        err = fmt.Errorf("%s error", "test")
        fmt.Println(err) //test error
    }
    
  4. error可以使用在函数返回
    package main
    
    import (
        "errors"
        "fmt"
    )
    
    func main() {
        if v, err := div(2, 0); err != nil {
            fmt.Println("error:", err)
        } else {
            fmt.Println("2/0 = ", v)
        }
    }
    
    func div(a, b int) (rs int, err error) {
        if b == 0 {
            err = errors.New("division by zero")
            return
        }
        return a / b, nil
    }
    
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,717评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,501评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,311评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,417评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,500评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,538评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,557评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,310评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,759评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,065评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,233评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,909评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,548评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,172评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,420评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,103评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,098评论 2 352

推荐阅读更多精彩内容

  • 使用定时器t := time.NewTicker(1 * time.Second)// 第一种方式for { ...
    qishuai阅读 915评论 0 2
  • 一、数据类型转换 https://studygolang.com/articles/10838 package m...
    蓓蓓的万能男友阅读 1,071评论 0 1
  • fmt格式化字符串 格式:%[旗标][宽度][.精度][arg索引]动词旗标有以下几种:+: 对于数值类型总是输出...
    皮皮v阅读 1,095评论 0 3
  • Golang是我最喜欢的一门语言,它简洁、高效、易学习、开发效率高、还可以编译成机器码… 虽然它一出世,就饱受关注...
    盘木阅读 3,548评论 0 7
  • 羡慕别人的七夕,有蜡烛、有拥抱、有鲜花、有掌声。 再看我的七夕,只有可怜的七毛七 我内心一哂,终究是太高估这个男人...
    baicaochunsheng阅读 220评论 0 0