01 Go 语言原理和基础实战(上)| 青训营笔记

85 阅读2分钟

这是我参与「第五届青训营」伴学笔记创作活动的第一天。
这一篇是之前学习过的课程笔记总结。记录自己学习过程。

知识要点

  1. Go语言上手
    • 简介
    • 开发环境
    • 基础语法
  2. 实战演练(见下)

知识点详解

简介

Golang 特点:

  1. 高性能,高并发
  2. 语法简单、学习曲线平缓
  3. 丰富的标准库
  4. 完善的工具链
  5. 静态编译
  6. 快速编译
  7. 跨平台
  8. 垃圾回收

开发环境

  1. 安装golang,可以使用的链接如下:

go.dev/
studygolang.com/dl
goproxy.cn/

  1. 安装开发环境(配置集成开发环境) 1)VS Code,GoLand(本人使用VSCode)

  2. 其他环境(基于云的开发环境)

gitpod.io/#github.com… (如果有Github账号使用)

本人使用的方法:(VS Code进行安装,参考博客4)

一些个人安装实践中的建议:

  1. 使用Vs Code安装环境时候把 Code Runner这一插件安装好,这样可以使用快捷键直接编译运行(初学快速上手便于减少环境配置文件撰写)
  2. 提前把一些默认配置文件撰写好,有需要时候才定制:如launch.json. 本人launch.json默认设置如下: (加入方式为左上角文件->首选项->设置->找到右上角(文件图标:打开设置(json)),加入下面一段进入JSON文件配置即可,如下图所示)

1676089298537.jpg

"launch": {
        "version": "0.2.0",
        "configurations": [ 
            { //go 语言的launch设置
                "name": "GoLaunch",
                "type": "go",
                "request": "launch",
                "mode": "debug",
                //"remotePath": "",
                "port": 2345,
                "host": "127.0.0.1",
                "program": "${fileDirname}",
                "env": {
                    "GOPATH": "E:/gowork"
                },
                "args": [],
                "showLog": true
            },
            //其他语言的launch设置 
            }

基础语法

(这里假定对一种编程语言,如C/C++/Java/Python 基本语法比较熟悉)

  1. **一个简单的Hello world **
package main

import (
	"fmt"
)

func main() {
	fmt.Println("hello world")
}

(形式上类似C++和Python,这里不仔细讲解)

  1. 变量
  • 强类型语言(与C语言一样)
  • 常见的变量有:字符串,整数,浮点,布尔类型。
  • 注意字符串是内置类型,支持一些运算(如下g := a + "foo",拼接使用“+”)
  • 运算符优先级与C类似。
  • 变量的声明
    第一种:自动推导(与c++ auto类似,如下var a = "initial"
    第二种:变量名 := 变量值
  • 常量 可以根据上下文自动确定类型,没有确定类型。
package main

import (
	"fmt"
	"math"
)

func main() {

	var a = "initial"

	var b, c int = 1, 2

	var d = true

	var e float64

	f := float32(e)

	g := a + "foo"
	fmt.Println(a, b, c, d, e, f) // initial 1 2 true 0 0
	fmt.Println(g)                // initialapple

	const s string = "constant"
	const h = 500000000
	const i = 3e20 / h
	fmt.Println(s, h, i, math.Sin(h), math.Sin(i))
}
  1. 条件判断 if/else
    如下,类似C 但是: 1.没有对条件的小括号;2.对大括号要求写在if/else 一行(有一定格式要求)
package main

import "fmt"

func main() {

	if 7%2 == 0 {
		fmt.Println("7 is even")
	} else {
		fmt.Println("7 is odd")
	}

	if 8%4 == 0 {
		fmt.Println("8 is divisible by 4")
	}

	if num := 9; num < 0 {
		fmt.Println(num, "is negative")
	} else if num < 10 {
		fmt.Println(num, "has 1 digit")
	} else {
		fmt.Println(num, "has multiple digits")
	}
}
  1. 循环 只有for循环
    注意: 1.死循环不需要写条件体; 2.其他类似C语言写法(注意条件体的括号和省略形式)
package main

import "fmt"

func main() {

	i := 1
	for {
		fmt.Println("loop")
		break
	}
	for j := 7; j < 9; j++ {
		fmt.Println(j)
	}

	for n := 0; n < 5; n++ {
		if n%2 == 0 {
			continue
		}
		fmt.Println(n)
	}
	for i <= 3 {
		fmt.Println(i)
		i = i + 1
	}
}
  1. Switch分支
    注意: 1.与C类似,可以不用写break 2.比C更加强大,判断条件不需要一定是整型。
package main

import (
	"fmt"
	"time"
)

func main() {

	a := 2
	switch a {
	case 1:
		fmt.Println("one")
	case 2:
		fmt.Println("two")
	case 3:
		fmt.Println("three")
	case 4, 5:
		fmt.Println("four or five")
	default:
		fmt.Println("other")
	}

	t := time.Now()
	switch {
	case t.Hour() < 12:
		fmt.Println("It's before noon")
	default:
		fmt.Println("It's after noon")
	}
}
  1. 数组和切片
    前者形式和C语言基本一致(但是由于固定长度,没有办法进行更改)
    后者在实际场景中应用更多(能够动态调整),感觉形式类似C++中的vector和string,以及使用方法与Python切片一致。
package main

import "fmt"

func main() {

	var a [5]int
	a[4] = 100
	fmt.Println("get:", a[2])
	fmt.Println("len:", len(a))

	b := [5]int{1, 2, 3, 4, 5}
	fmt.Println(b)

	var twoD [2][3]int
	for i := 0; i < 2; i++ {
		for j := 0; j < 3; j++ {
			twoD[i][j] = i + j
		}
	}
	fmt.Println("2d: ", twoD)
}
package main

import "fmt"

func main() {

	s := make([]string, 3)
	s[0] = "a"
	s[1] = "b"
	s[2] = "c"
	fmt.Println("get:", s[2])   // c
	fmt.Println("len:", len(s)) // 3

	s = append(s, "d")
	s = append(s, "e", "f")
	fmt.Println(s) // [a b c d e f]

	c := make([]string, len(s))
	copy(c, s)
	fmt.Println(c) // [a b c d e f]

	fmt.Println(s[2:5]) // [c d e]
	fmt.Println(s[:5])  // [a b c d e]
	fmt.Println(s[2:])  // [c d e f]

	good := []string{"g", "o", "o", "d"}
	fmt.Println(good) // [g o o d]
}
  1. 哈希字典-map
    与C++ unordered_map用法对应,示例如下
package main

import "fmt"

func main() {
	m := make(map[string]int)
	m["one"] = 1
	m["two"] = 2
	fmt.Println(m)           // map[one:1 two:2]
	fmt.Println(len(m))      // 2
	fmt.Println(m["one"])    // 1
	fmt.Println(m["unknow"]) // 0

	r, ok := m["unknow"]
	fmt.Println(r, ok) // 0 false

	delete(m, "one")

	m2 := map[string]int{"one": 1, "two": 2}
	var m3 = map[string]int{"one": 1, "two": 2}
	fmt.Println(m2, m3)
}
  1. Range 遍历
    类似于Python解包操作+for
package main

import "fmt"

func main() {
	nums := []int{2, 3, 4}
	sum := 0
	for i, num := range nums {
		sum += num
		if num == 2 {
			fmt.Println("index:", i, "num:", num) // index: 0 num: 2
		}
	}
	fmt.Println(sum) // 9

	m := map[string]string{"a": "A", "b": "B"}
	for k, v := range m {
		fmt.Println(k, v) // b 8; a A
	}
	for k := range m {
		fmt.Println("key", k) // key a; key b
	}
}
  1. 函数
    这里和C语言类似。
    注意:1.变量类型后置;2.支持多个值返回,一般最后返回错误信息。
package main

import "fmt"

func add(a int, b int) int {
	return a + b
}

func add2(a, b int) int {
	return a + b
}

func exists(m map[string]string, k string) (v string, ok bool) {
	v, ok = m[k]
	return v, ok
}

func main() {
	res := add(1, 2)
	fmt.Println(res) // 3

	v, ok := exists(map[string]string{"a": "A"}, "a")
	fmt.Println(v, ok) // A True
}
  1. 指针
    这里支持操作比较有限,主要修改传入函数内的值修改。
package main

import "fmt"

func add2(n int) {
	n += 2
}

func add2ptr(n *int) {
	*n += 2
}

func main() {
	n := 5
	add2(n)
	fmt.Println(n) // 5
	add2ptr(&n)
	fmt.Println(n) // 7
}
  1. 结构体和结构体方法
    类似于C++语言结构体 (struct/class) 注意成员函数对应普通函数,代码区别。
package main

import "fmt"

type user struct {
	name     string
	password string
}

func main() {
	a := user{name: "wang", password: "1024"}
	b := user{"wang", "1024"}
	c := user{name: "wang"}
	c.password = "1024"
	var d user
	d.name = "wang"
	d.password = "1024"

	fmt.Println(a, b, c, d)                 // {wang 1024} {wang 1024} {wang 1024} {wang 1024}
	fmt.Println(checkPassword(a, "haha"))   // false
	fmt.Println(checkPassword2(&a, "haha")) // false
}

func checkPassword(u user, password string) bool {
	return u.password == password
}

func checkPassword2(u *user, password string) bool {
	return u.password == password
}
package main

import "fmt"

type user struct {
	name     string
	password string
}

func (u user) checkPassword(password string) bool { 
	return u.password == password
}

func (u *user) resetPassword(password string) {
	u.password = password
}

func main() {
	a := user{name: "wang", password: "1024"}
	a.resetPassword("2048")
	fmt.Println(a.checkPassword("2048")) // true
}
  1. 错误处理
package main

import (
	"errors"
	"fmt"
)

type user struct {
	name     string
	password string
}

func findUser(users []user, name string) (v *user, err error) {
	for _, u := range users {
		if u.name == name {
			return &u, nil
		}
	}
	return nil, errors.New("not found")
}

func main() {
	u, err := findUser([]user{{"wang", "1024"}}, "wang")
	if err != nil {
		fmt.Println(err)
		return
	}
	fmt.Println(u.name) // wang

	if u, err := findUser([]user{{"wang", "1024"}}, "li"); err != nil {
		fmt.Println(err) // not found
		return
	} else {
		fmt.Println(u.name)
	}
}
  1. 字符串操作和格式化
    为了不简单罗列,直接展示相关代码如下
    这类操作建议在使用时多查多用多练习,从而逐渐熟练。
package main

import (
	"fmt"
	"strings"
)

func main() {
	a := "hello"
	fmt.Println(strings.Contains(a, "ll"))                // true
	fmt.Println(strings.Count(a, "l"))                    // 2
	fmt.Println(strings.HasPrefix(a, "he"))               // true
	fmt.Println(strings.HasSuffix(a, "llo"))              // true
	fmt.Println(strings.Index(a, "ll"))                   // 2
	fmt.Println(strings.Join([]string{"he", "llo"}, "-")) // he-llo
	fmt.Println(strings.Repeat(a, 2))                     // hellohello
	fmt.Println(strings.Replace(a, "e", "E", -1))         // hEllo
	fmt.Println(strings.Split("a-b-c", "-"))              // [a b c]
	fmt.Println(strings.ToLower(a))                       // hello
	fmt.Println(strings.ToUpper(a))                       // HELLO
	fmt.Println(len(a))                                   // 5
	b := "你好"
	fmt.Println(len(b)) // 6
}
package main

import "fmt"

type point struct {
	x, y int
}

func main() {
	s := "hello"
	n := 123
	p := point{1, 2}
	fmt.Println(s, n) // hello 123
	fmt.Println(p)    // {1 2}

	fmt.Printf("s=%v\n", s)  // s=hello
	fmt.Printf("n=%v\n", n)  // n=123
	fmt.Printf("p=%v\n", p)  // p={1 2}
	fmt.Printf("p=%+v\n", p) // p={x:1 y:2}
	fmt.Printf("p=%#v\n", p) // p=main.point{x:1, y:2}

	f := 3.141592653
	fmt.Println(f)          // 3.141592653
	fmt.Printf("%.2f\n", f) // 3.14
}

实例

  1. JSON文件处理
    一些注释:
    1)载入包是 encoding/json。
    2)struct 结构体第一个字母为大写。
    3)json.Marshal 序列化。
    4)序列化后变量注意用string转换。
    5)json.Unmarshal 反序列化。
    6)输出小写注意加一个json tag, 如下面的json:"age"
package main

import (
	"encoding/json"
	"fmt"
)

type userInfo struct {
	Name  string
	Age   int `json:"age"`
	Hobby []string
}

func main() {
	a := userInfo{Name: "wang", Age: 18, Hobby: []string{"Golang", "TypeScript"}}
	buf, err := json.Marshal(a)
	if err != nil {
		panic(err)
	}
	fmt.Println(buf)         // [123 34 78 97...]
	fmt.Println(string(buf)) // {"Name":"wang","age":18,"Hobby":["Golang","TypeScript"]}

	buf, err = json.MarshalIndent(a, "", "\t")
	if err != nil {
		panic(err)
	}
	fmt.Println(string(buf))

	var b userInfo
	err = json.Unmarshal(buf, &b)
	if err != nil {
		panic(err)
	}
	fmt.Printf("%#v\n", b) // main.userInfo{Name:"wang", Age:18, Hobby:[]string{"Golang", "TypeScript"}}
}
  1. 时间处理
    一些注释:
    1)time.Now 获取当前时间。
    2)time.Data 获取带时序时间。
    3).Sub 得到时间相减的时间段。
    4)Format 格式化到时间字符串,注意被格式化的字符串格式。
    5)time.Unix 获取时间戳。
package main

import (
	"fmt"
	"time"
)

func main() {
	now := time.Now()
	fmt.Println(now) // 2022-03-27 18:04:59.433297 +0800 CST m=+0.000087933
	t := time.Date(2022, 3, 27, 1, 25, 36, 0, time.UTC)
	t2 := time.Date(2022, 3, 27, 2, 30, 36, 0, time.UTC)
	fmt.Println(t)                                                  // 2022-03-27 01:25:36 +0000 UTC
	fmt.Println(t.Year(), t.Month(), t.Day(), t.Hour(), t.Minute()) // 2022 March 27 1 25
	fmt.Println(t.Format("2006-01-02 15:04:05"))                    // 2022-03-27 01:25:36
	diff := t2.Sub(t)
	fmt.Println(diff)                           // 1h5m0s
	fmt.Println(diff.Minutes(), diff.Seconds()) // 65 3900
	t3, err := time.Parse("2006-01-02 15:04:05", "2022-03-27 01:25:36")
	if err != nil {
		panic(err)
	}
	fmt.Println(t3 == t)    // true
	fmt.Println(now.Unix()) // 1648738080
}
  1. 数字解析
    一些注释:
    1)ParseInt 解析字符串为int类型;第二个表示解析出来的进制,全零需要自动推测;第三个表示解析得到结果所占用位数。
    2)ParseFloat 解析字符串为float类型;第二个表示解析得到结果所占用位数。
    3)Atoi 类似C语言atoi,如果有误则转化为0,返回错误信息。
package main

import (
	"fmt"
	"strconv"
)

func main() {
	f, _ := strconv.ParseFloat("1.234", 64)
	fmt.Println(f) // 1.234

	n, _ := strconv.ParseInt("111", 10, 64)
	fmt.Println(n) // 111

	n, _ = strconv.ParseInt("0x1000", 0, 64)
	fmt.Println(n) // 4096

	n2, _ := strconv.Atoi("123")
	fmt.Println(n2) // 123

	n2, err := strconv.Atoi("AAA")
	fmt.Println(n2, err) // 0 strconv.Atoi: parsing "AAA": invalid syntax
}
  1. 进程信息
    一些注释:
    1)下面是对main函数进行传参,os.Args获取参数信息
    第一个表示当前文件,后面表示执行所传入参数
    2)os.Getenv/Setenv 获得环境变量或者设置环境变量
    3)exec.Command 启动子进程,获得其相应的输入输出。
package main

import (
	"fmt"
	"os"
	"os/exec"
)

func main() {
	// go run example/20-env/main.go a b c d
	fmt.Println(os.Args)           // [/var/folders/8p/n34xxfnx38dg8bv_x8l62t_m0000gn/T/go-build3406981276/b001/exe/main a b c d]
	fmt.Println(os.Getenv("PATH")) // /usr/local/go/bin...
	fmt.Println(os.Setenv("AA", "BB"))

	buf, err := exec.Command("grep", "127.0.0.1", "/etc/hosts").CombinedOutput()
	if err != nil {
		panic(err)
	}
	fmt.Println(string(buf)) // 127.0.0.1       localhost
}

参考资料和补充阅读

  1. 课程地址:juejin.cn/course/byte…
  2. 课程源代码:github.com/wangkechun/…
  3. 课程参考书籍:前言 · Go语言圣经 (studygolang.com)
  4. Go语言环境安装 用 Goland和 VS Code 搭建Go 语言开发环境

欢迎评论指正和补充