这是我参与「第三届青训营 -后端场」笔记创作活动的的第1篇笔记
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}
结构体方法,不同于一般的函数,是运用结构体对象的时候可以用“.”来调用的方法。
错误处理
err
nil
字符串操作
字符串格式化
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代码。
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)
}
}
执行结果:
3.3 SOCKS5代理
这个完全是知识盲区,磨蹭到晚上九点才开始慢慢学。放到linux虚拟机上跑了一下。 工作原理: