GO语言基础与实战 | 青训营笔记

58 阅读3分钟

Go语言基础

学习GO语言的基本语法

变量

Go语言支持多种变量声明方式,Go是一个强类型语言

// 直接声明,不指定变量类型
x := 5
word := "Hello"

// var声明方法
// 只定义 var v_name v_type
var word string
word = "Hello"
// 定义的同时声明
var word string = "Hello"
var word1, word2 string = "Hello", "World"

循环

Go语言只有for循环,使用方法类似于C++

for j := 0; j < 10; j++ {
    fmt.Println(j)
}

条件语句

Go语言有if条件语句和switch条件语句。 If条件语句类似于C++, 但是switch语句有更强大的功能,可以进行比较复杂的判断

// 分别使用if和switch实现判断一个数字是否大于0,大于10
x := 15
if x < 0 {
        fmt.Println("x is a negative number")
} else {
        if x > 10 {
                fmt.Println("x is larger than 10")
        } else {
                fmt.Println("0 <= x <= 10")
        }
}

switch {
case x < 0:
        fmt.Println("x is a negative number")
case x > 10:
        fmt.Println("x is larger than 10")
default:
        fmt.Println("0 <= x <= 10")
}

可以看出switch语句可以较为简洁的实现一些if-else的功能

数组

Go 语言数组声明需要指定元素类型及元素个数,语法格式如下:

var variable_name [SIZE] variable_type

//eg
var word [10] string

// 也可以在声明时候定义
word := [10]string{"Hello", "World"}

切片

Go 语言切片是对数组的抽象。

Go 数组的长度不可改变,在特定场景中这样的集合就不太适用,Go 中提供了一种灵活,功能强悍的内置类型切片("动态数组"),与数组相比切片的长度是不固定的,可以追加元素,在追加时可能使切片的容量增大。

定义切片

通过声明一个未定长数组来定义切片

var word []string

或者使用make()函数来创建

var slice1 []type = make([]type, len)

// 可以简写为
slice1 := make([]type, len)

切片初始化

如果要进行扩容,需要使用append, 不同于python中的append

var word []string
word = append(word, "World")
// 切片有了内容以后通过下标访问直接修改
word[0] = "Hello"

// 直接初始化
good := []string{"g", "o", "o", "d"}

Map

Map 是一种无序的键值对的集合。Map 最重要的一点是通过 key 来快速检索数据,key 类似于索引,指向数据的值。 Map可以迭代访问,但是Map是完全无序的

定义Map

可以使用make()函数来定义,也可以使用map关键字

/* 声明变量,默认 map 是 nil */
var map_variable map[key_data_type]value_data_type

/* 使用 make 函数 */
map_variable := make(map[key_data_type]value_data_type)

使用

m := make(map[string]int)
m["one"] = 1
m["two"] = 2

// 索引元素 返回两个值,分别是value,和是否存在
value, is_exist := m["one"]

// 删除元素
delete(m, "one")

Range

Go 语言中 range 关键字用于 for 循环中迭代数组(array)、切片(slice)、通道(channel)或集合(map)的元素。在数组和切片中它返回元素的索引和索引对应的值,在集合中返回 key-value 对。

for 循环的 range 格式可以对 slice、map、数组、字符串等进行迭代循环

// 访问数组返回index和对应位置的数值
nums := []int{2, 3, 4}
for i, num := range nums {
    sum += num
    if num == 2 {
        fmt.Println("index:", i, "num:", num) // index: 0 num: 2
    }
    
// 访问key-value对
m := map[string]string{"a": "A", "b": "B"}
for k, v := range m {
    fmt.Println(k, v) // b 8; a A
}

函数

在Go语言中,参数传递类似于C++,默认情况下是值传递,如果函数需要对传入的参数进行修改,需要使用到指针,传入变量的地址。

// 函数的定义
func func_name(parameter list) return_type {
    func body;
}

//eg
func max(num1, num2 int) int {
    var result int
    if (num1 > num2) {
        result = num1
    } else {
        result = num2
    }
    return result
}

在Go中支持函数返回多个值

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

指针

用于函数传参修改变量。和C++类似

func swap(num1 *int, num2 *int) {
    temp := *num1
    *num1 = *num2
    *num2 = temp
}

结构体

给具有多种相同属性的类型创建一个结构体

// 定义结构体
type struce_name struct {
    member definition
}

type user struct {
    name     string
    password string
}

// 声明格式
variable_name := structure_variable_type {value1, value2...valuen}
variable_name := structure_variable_type { key1: value1, key2: value2..., keyn: valuen}

// 通过.操作访问结构体成员
a := user("TAO", "12138")
a.password = "12210"

方法

可以给结构体定义方法,类似于成员函数

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

// 需要修改结构体内部变量的时候传指针
func (u *user) resetPassword(password string) {
    u.password = password
}

字符串

Go提供了处理字符串的很多方法

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

defer

Go实战

猜数字

首先生成随机数,使用rand,并通过time设置随机数种子

maxNum := 100
rand.Seed(time.Now().UnixNano())
secretNum :=  rand.Intn(maxNum)

为了使得用户可以一直进行猜测直到猜对数字,所以设置for循环,直到数字正确break

reader := bufio.NewReader(os.Stdin)
for {
    input, err := reader.ReadString('\n')
    if err != nil {
            fmt.Println("An error occured while reading input. Please try again", err)
            continue
    }
    input = strings.Trim(input, "\r\n")

    guess, err := strconv.Atoi(input)
    if err != nil {
        fmt.Println("Invalid input. Please enter an integer value")
        continue
    }
    fmt.Println("You guess is", guess)
    if guess > secretNum {
        fmt.Println("Your guess is bigger than the secret number. Please try again")
    } else if guess < secretNum {
    fmt.Println("Your guess is smaller than the secret number. Please try again")
    } else {
        fmt.Println("Correct, you Legend!")
        break
    }
}

词典

主要学习使用Go语言发起http请求,并通过处理json来清洗返回结果

package main

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

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

type DictResponse struct {
	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) {
	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("Connection", "keep-alive")
	req.Header.Set("DNT", "1")
	req.Header.Set("os-version", "")
	req.Header.Set("sec-ch-ua-mobile", "?0")
	req.Header.Set("User-Agent", "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/99.0.4844.51 Safari/537.36")
	req.Header.Set("app-name", "xy")
	req.Header.Set("Content-Type", "application/json;charset=UTF-8")
	req.Header.Set("Accept", "application/json, text/plain, */*")
	req.Header.Set("device-id", "")
	req.Header.Set("os-type", "web")
	req.Header.Set("X-Authorization", "token:qgemv4jr1y38jyq6vhvi")
	req.Header.Set("Origin", "https://fanyi.caiyunapp.com")
	req.Header.Set("Sec-Fetch-Site", "cross-site")
	req.Header.Set("Sec-Fetch-Mode", "cors")
	req.Header.Set("Sec-Fetch-Dest", "empty")
	req.Header.Set("Referer", "https://fanyi.caiyunapp.com/")
	req.Header.Set("Accept-Language", "zh-CN,zh;q=0.9")
	req.Header.Set("Cookie", "_ym_uid=16456948721020430059; _ym_d=1645694872")
	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)
	}
	if resp.StatusCode != 200 {
		log.Fatal("bad StatusCode:", resp.StatusCode, "body", string(bodyText))
	}
	var dictResponse DictResponse
	err = json.Unmarshal(bodyText, &dictResponse)
	if err != nil {
		log.Fatal(err)
	}
	fmt.Println(word, "UK:", dictResponse.Dictionary.Prons.En, "US:", dictResponse.Dictionary.Prons.EnUs)
	for _, item := range dictResponse.Dictionary.Explanations {
		fmt.Println(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)
}