1.Go语言简介
1.1什么是Go语言
- 高性能、高并发 Go的速度也非常快,几乎和C或C 程序一样快,且能够快速制作程序。Go支持面向对象,而且具有真正的闭包(closures)和反射 (reflection)等功能。
语言层面支持并发,这个就是Go最大的特色,天生的支持并发,Go就是基因里面支持的并发,可以充分的利用多核,很容易的使用并发。
-
语法简单、学习曲线平滑 简单易学,Go语言的作者都有C的基因,那么Go自然而然就有了C的基因,那么Go关键字是25个,但是表达能力很强大,几乎支持大多数你在其他语言见过的特性:继承、重载、对象等。
-
丰富的标准库 丰富的标准库,Go目前已经内置了大量的可靠并且功能完善的标准库,在大多数情况下不需要去借助第三方库进行开发,并且标准库拥有着很高的稳定性和兼容性保障,还能持续享受语言迭代带来的性能优化,这是第三方库所不具备的。
-
完善的工具链 Go语言在诞生之初就有着丰富的工具链,无论是编译、代码格式化、错误检查、帮助文档、包管理、代码补充提示这些都有对应的工具,Go语言还内置了完整的单元测试框架,能够进行性能测试、代码覆盖率、数值竞争检测、代码优化等,这些都是能够保证代码能够稳定运行的必备利器。
-
静态链接 Go语言的默认连接方式就是静态链接,只需要拷贝编译链接后的唯一一个文件,而不需要附加任何东西就能部署运行,在线上的容器环境下运行,镜像体积可以控制地非常小,部署非常方便快捷。
-
快速编译 Go语言有着静态语言中几乎最快的编译速度。
-
跨平台 Go语言能够在Windows,Linux,Max,Mac等常见的操作系统中运行,也能够用来开发Android、IOS等软件,也能够在其它设备上运行,包括路由器、树莓派,因为它有着交叉编译特性。
-
垃圾回收 Go语言是一门带垃圾回收的语言,和Java类似,你无需考虑内存的分配释放。
2.入门
2.1开发环境
安装Golang
VSCode
blog.csdn.net/ZHAOJUNWEI0… (均为转载)
2.2基础语法
package main //包名称
import "fmt" //导入标准输出包
func main() {
fmt.Println("Hello World!") //调用函数来输出
}
不难看出Go在标记包,导包,以及输出方面均类似与Java。
- 变量 Go语言是一种强类型语言,每一个变量都有自己的变量类型。 常见的类型包括字符串、整数、浮点型、布尔型。
两种变量声明方式,标准格式以及简短格式
标准格式(var可以推导类型,但也可以用type来写死)
var name type
简短格式(通过表达式来推导)
name := 表达式
-
常量 将变量中的var改成const即可,值得一提的是这里面的常量没有确定的类型,而是通过使用的上下文来确定类型。
-
if else 与C的 if else 非常类似,不同点在与 if 面的判断时没有括号的,这样的话 if 后面就必须有大括号。
-
循环 在这里面只有for循环,同时这个for是加强过的,它能进行之前C里面while的操作。
-
switch
同样省去了switch后面的括号,还有一个不同之处就是不需要加break。这个switch也是经过加强的,它可以不再接受更多类型的变量,并且可以在switch里面再进行条件判断。
- 数组
逻辑上并没有什么不同。
- 切片 切片是一个可变长度的数组,我们有make来创造一个切片。下面进行演示:
s := make([]string, 3)
s[0] = "a"
s[1] = "b"
s[2] = "c"
s = append(s, "d", "f") //使用append来追加数组元素
- map
m := make(map[string]int) //key的类型是string,value的类型是int
m["one"] = 1
m["two"] = 2
fmt.Println(m) // map[one:1 two:2]
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}
- range 对于数组和map,range可以快速遍历.
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
}
-
函数 和其他语言不一样的是,变量的类型是后置的,并且可以返回多个值(第一个值是结果,第二个值是错误信息).
-
指针 Golang对于指针的使用非常少,就是最经典的函数传参使用指针类型改变原值.
-
结构体 和C基本类似
type user struct{
name string
password string
}
- 结构体方法 比较类似于类成员方法.
func (u user) checkPassword(password string) bool {
return u.password = password
}
- 错误处理 不同于Java中的抛异常,Golong里面的错误处理类似于C中的处理方式.但优于C的点就在于它可以返回多个值,所以再返回结构的同时,也能返回错误.
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)
}
}
- 字符串操作
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
- 字符串格式化输出
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
}
- JSON处理
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"}}
}
- 时间处理
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
}
- 数字解析
package main
import (
"fmt"
"strconv"
)
func main() {
f, _ := strconv.ParseFloat("1.234", 64)
fmt.Println(f) // 1.234
n, _ := strconv.ParseInt("111", 10, 64)//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
}
- 进程信息
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
}
3.简单实战
3.1. 猜字谜
package main
import (
"bufio" //读写缓冲区
"fmt"
"math/rand"
"os" //系统及操作
"strconv" //数字解析
"strings"
"time"
)
func main() {
maxNum := 50
rand.Seed(time.Now().UnixNano()) //用现在1时间戳来当随机种子(常规操作)
randNum := rand.Intn(maxNum)
fmt.Println("The number is", randNum)
fmt.Println("Please input your guess")
reader := bufio.NewReader(os.Stdin) //new出了一个读写缓冲区
//循环判断
for {
input, err := reader.ReadString('\n') //读取一行字符串
if err != nil {
fmt.Println("An error occured while reading input. Please try again", err)
//return
continue
}
input = strings.TrimSuffix(input, "\r\n") //将尾部的转义字符去除
guess, err := strconv.Atoi(input) //判断其是否为数字
if err != nil {
fmt.Println("Invalid input. Please enter an integer value")
//return
continue
}
fmt.Println("You guess is", guess)
if guess > randNum {
fmt.Println("Your guess is bigger.Please try again:")
} else if guess < randNum {
fmt.Println("Your guess is smaller.Please try again:")
} else {
fmt.Println("Your guess is right.")
break
}
}
}
结果如下:
3.2. 在线字典
这段代码是考过来的,因为在火狐上导过去的cURL转换出的代码总是无法发送请求,而我又不会跳外网,没办法用谷歌,所以只是了解了以下大概的操作过程.之后想办法翻外网,再次处理这个问题.
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)
}
结果如下: