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)
}