Go语言上手-基础语言 | 青训营笔记

98 阅读7分钟

这是我参与「第三届青训营 -后端场」笔记创作活动的的第1篇笔记

image.png

image.png

1.1 Go语言特性

高性能、高并发,语法简单,标准库丰富,工具链完善,静态链接,快速编译,跨平台,垃圾回收。

2.1 开发环境-安装

【Go 语言原理与实践学习资料】第三届字节跳动青训营-后端专场juejin.cn/post/709372…

2.2 基础语法

hello world.go

package main//main包是程序的入口包,表示此文件是程序的入口文件

import (
    "fmt"
)

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

运行 go run go build

变量声明

go语言中常见的变量类型:字符串,整数,浮点数,布尔型

变量间比较 用==可以直接比较

var a = "hello"

var b,c int = 1,2

var d = true

var e float64

f := float32(e)

g := a + "foo" //字符串可以直接相连

const s string = "constant"//常量

const h = 50000000

const i = 3e20/h

math.Sin(h)

if else

if条件没有括号

里面的内容必须换行

for循环

i:=1 for{//死循环 xx } for xx;xx;xx {//三段式任何一段都可以省略

switch case

switch a {//不需要加括号
case 1:
    xxx
case 2:
    xxx
default:
    xxx
}

与其他语言不同,每个case中不需要写break也能跳过其他分支。

数组

var a [5]int

工程中很少用,因为长度不可变。

slice

可变长度的数组

s := make([]string, 3)
s[0] = a
s[1] = b
s[2] = c
s = append(s,"d")
s = append(s, "e", "f")

c := make([]string, len(s))//复制一个slice数组
copy(c,s)

fmt.Println(s[2:5])
fmt.Println(s[:5])
fmt.Println(s[2:])

good := []string{"g","o","o","d"}

map

其他语言中的hash或者字典,完全无序,不会按插入顺序输出

m := make(map[string]int)//string是key的类型,ing是value的类型
m["one"] = 1
m["two"] = 2
fmt.Println(len(m))//2
fmt.Println(m)//map[one:1 two:2]
fmt.Println("one")//通过key读取value
fmt.Println("unknown")// 返回0

//ok用来获取是否存在这个key
r, ok := m["unknown"]
fmt.Println(r, ok)// 0 false

delete(m,"one")

range

range可以用来快速遍历slice或者map,具体代码如下:

for i,num := range nums {

若遍历slice, 第一个值是索引,第二个是value

若遍历map, 第一个值是key,第二个是value

用不到的用_,只写一个为第一个

函数

几乎所有函数都返回两个值,第一个是真正的返回值,而第二个是错误信息。

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

结构体

结构体,带类型的字段的集合

type user struct {
    name string
    password string
}

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

结构体方法,不同于一般的函数,是运用结构体对象的时候可以用“.”来调用的方法。 image.png

错误处理

err

nil

字符串操作

image.png

字符串格式化

image.png

json处理

定义与结构体差不多,但是每个类型第一个字母大写,代表公开字段,然后对结构体序列化(json.Marshal),生成一个二进制字段

package main

import (
	"encoding/json"
	"fmt"
)

type userInfo struct {
	Name  string
	Age   int `json:"age"`//加了一个json的tag,序列化后输出时可以显示小写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"}}
}

时间处理

"time"包

        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
        //格式化,用一个特定的时间格式化,不用写yy-mm啥的
	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

数字解析

"strconv"包string convert

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

	n, _ := strconv.ParseInt("111", 10, 64) 
        //第二个数字代表进制,8,10,16,0为自动
        //第三个数字是转换后的精度 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

进程信息

"os""os/exec"

// 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]
        //输出的args长度为5,第一个是路径和执行的文件名,后面4个是输入的参数
        
        //获取环境变量
        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

实战

3.1 猜谜游戏

获取输入字符串时要去掉最后的\n, 但windows系统需要去掉\r\n

3.2 字典

课程示例中使用了彩云小译 - 在线翻译 (caiyunapp.com)

两个自动生成代码的网站:

在浏览器中找到检查,点击“网络”,找到最新的 dict (POST) 请求,右键复制cURL,至网站curlconverter.com/#go,可以自动生成r… body的go代码。 image.png preview(预览)里的json复制到这个网站自动生成response结构体:oktools.net/json2go

全部代码如下:

package main

import (
	"bytes"
	"encoding/json"
	"fmt"
	"io/ioutil"
	"log"
	"net/http"
	"os"
)

type DictRequest struct { //request
	TransType string `json:"trans_type"`
	Source    string `json:"source"`
	UserID    string `json:"user_id"`
}

//将网页上复制的preview里的json复制到这个网站自动生成结构体:https://oktools.net/json2go
type DictResponse struct { //response
	Rc   int `json:"rc"`
	Wiki struct {
		KnownInLaguages int `json:"known_in_laguages"`
		Description     struct {
			Source string      `json:"source"`
			Target interface{} `json:"target"`
		} `json:"description"`
		ID   string `json:"id"`
		Item struct {
			Source string `json:"source"`
			Target string `json:"target"`
		} `json:"item"`
		ImageURL  string `json:"image_url"`
		IsSubject string `json:"is_subject"`
		Sitelink  string `json:"sitelink"`
	} `json:"wiki"`
	Dictionary struct {
		Prons struct {
			EnUs string `json:"en-us"`
			En   string `json:"en"`
		} `json:"prons"`
		Explanations []string      `json:"explanations"`
		Synonym      []string      `json:"synonym"`
		Antonym      []string      `json:"antonym"`
		WqxExample   [][]string    `json:"wqx_example"`
		Entry        string        `json:"entry"`
		Type         string        `json:"type"`
		Related      []interface{} `json:"related"`
		Source       string        `json:"source"`
	} `json:"dictionary"`
}

func query(word string) {
	//生成Request body,先在浏览器中找到cURL,用这个网站https://curlconverter.com/#go
	client := &http.Client{}
	request := DictRequest{TransType: "en2zh", Source: word}
	buf, err := json.Marshal(request)
	if err != nil {
		log.Fatal(err)
	}
	var data = bytes.NewReader(buf)
	req, err := http.NewRequest("POST", "https://api.interpreter.caiyunai.com/v1/dict", data)
	if err != nil {
		log.Fatal(err)
	}
	req.Header.Set("Accept", "application/json, text/plain, */*")
	req.Header.Set("Accept-Language", "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6")
	req.Header.Set("Connection", "keep-alive")
	req.Header.Set("Content-Type", "application/json;charset=UTF-8")
	req.Header.Set("Origin", "https://fanyi.caiyunapp.com")
	req.Header.Set("Referer", "https://fanyi.caiyunapp.com/")
	req.Header.Set("Sec-Fetch-Dest", "empty")
	req.Header.Set("Sec-Fetch-Mode", "cors")
	req.Header.Set("Sec-Fetch-Site", "cross-site")
	req.Header.Set("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/100.0.4896.60 Safari/537.36 Edg/100.0.1185.29")
	req.Header.Set("X-Authorization", "token:qgemv4jr1y38jyq6vhvi")
	req.Header.Set("app-name", "xy")
	req.Header.Set("os-type", "web")
	req.Header.Set("sec-ch-ua", `" Not A;Brand";v="99", "Chromium";v="100", "Microsoft Edge";v="100"`)
	req.Header.Set("sec-ch-ua-mobile", "?0")
	req.Header.Set("sec-ch-ua-platform", `"Windows"`)
	resp, err := client.Do(req) //执行request
	if err != nil {
		log.Fatal(err)
	}
	defer resp.Body.Close() //main结束时执行
	bodyText, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		log.Fatal(err)
	}
	if resp.StatusCode != 200 {
		log.Fatal("bad StatusCode:", resp.StatusCode, "body", string(bodyText))
	}
	//解析response body
	var dictResponse DictResponse
	err = json.Unmarshal(bodyText, &dictResponse) //通过解开序列化数组得到response结构体
	if err != nil {
		log.Fatal(err)
	}
	fmt.Println(word, "UK:", dictResponse.Dictionary.Prons.En, "US:", dictResponse.Dictionary.Prons.EnUs)
	//循环打印explanations数组
	for _, item := range dictResponse.Dictionary.Explanations {
		fmt.Println(item)
	}
        //又打印了同义词,都在response里可以找到的
	for _, item := range dictResponse.Dictionary.Synonym {
        
		fmt.Print(item, " ")
	}
}

func main() {
	if len(os.Args) != 2 {
		fmt.Fprintf(os.Stderr, `usage: simpleDict WORD example: simpleDict hello`)
		os.Exit(1)
	}
	word := os.Args[1]
	query(word)
}

课后作业: 增加另一种翻译引擎的支持。

已完成,这里使用有道词典网页版,新增了一个query方法在main中调用。在有道词典中没找到POST的request,只有GET,于是使用GET request,在url中嵌入要查的单词,也同样达到了目的。方法代码如下,亲测有效:

func queryYoudao(word string) {
	client := &http.Client{}
	referer := "https://dict.youdao.com/result?word=" + word + "&lang=en"
	req, err := http.NewRequest("GET", "https://dict.youdao.com/suggest?num=5&ver=3.0&doctype=json&cache=false&le=en&q="+word, nil)
	if err != nil {
		log.Fatal(err)
	}
	req.Header.Set("Accept", "application/json, text/plain, */*")
	req.Header.Set("Accept-Language", "zh-CN,zh;q=0.9,en;q=0.8,en-GB;q=0.7,en-US;q=0.6")
	req.Header.Set("Connection", "keep-alive")
	req.Header.Set("Cookie", "OUTFOX_SEARCH_USER_ID=-1842890407@59.64.129.41; _ntes_nnid=65ff81b77a6d3171fcd3c5d1c1f49679,1589083821810; OUTFOX_SEARCH_USER_ID_NCOO=2134992473.278991; ___rl__test__cookies=1651917647130")
	req.Header.Set("Referer", referer)
	req.Header.Set("Sec-Fetch-Dest", "empty")
	req.Header.Set("Sec-Fetch-Mode", "cors")
	req.Header.Set("Sec-Fetch-Site", "same-origin")
	req.Header.Set("User-Agent", "Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/100.0.4896.60 Safari/537.36 Edg/100.0.1185.29")
	req.Header.Set("sec-ch-ua", `" Not A;Brand";v="99", "Chromium";v="100", "Microsoft Edge";v="100"`)
	req.Header.Set("sec-ch-ua-mobile", "?0")
	req.Header.Set("sec-ch-ua-platform", `"Windows"`)
	resp, err := client.Do(req)
	if err != nil {
		log.Fatal(err)
	}
	defer resp.Body.Close()
	bodyText, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		log.Fatal(err)
	}
	//解析response body
	var dictResponse2 DictResponse2
	err = json.Unmarshal(bodyText, &dictResponse2) //通过解开序列化数组得到response结构体
	if err != nil {
		log.Fatal(err)
	}
	fmt.Println("\n-----------------------\n", word, "by YouDao")
	//循环打印explanations数组
	for _, item := range dictResponse2.Data.Entries {
		fmt.Println(item)
	}

}

执行结果:

image.png

3.3 SOCKS5代理

这个完全是知识盲区,磨蹭到晚上九点才开始慢慢学。放到linux虚拟机上跑了一下。 工作原理:

image.png

image.png

image.png