Go语言基础语法 | 青训营笔记

215 阅读8分钟

这是我参与「第五届青训营 」伴学笔记创作活动的第 1 天

Go语言基础语法

简介

  • 什么是Go语言
    • 高性能、高并发
    • 语法简单、学习曲线平缓
    • 丰富的标准库
    • 完善的工具链
    • 静态链接
    • 快速编译
    • 跨平台
    • 垃圾回收

入门

开发环境

  • 安装Golang
  • 配置集成开发环境
  • 基于云的开发环境

基础语法

  • Hello World

    • 运行方式:直接运行、编译成二进制后运行
go run example/01-hello/main.go


go build example/01-hello/main.go
./main
  • 变量
    • 强类型语言
    • 常见类型:字符串、整数、浮点型、布尔型
    • 字符串:加号相加,等号比较
    • 变量声明的两种方式
    • 常量没有确定的类型,会根据使用的上下文自动确定类型
var a = "initial" //自动推导类型
var b,c int = 1, 2 //有需要也可以显式写出类型
const s string = "constant"
  • if else

    • if后面的判断语句不用括号
    • if要和第一个大括号在同一行
  • for

  • switch

    • 不需要用break防止程序在switch里执行多个分支
    • 在switch后面不加任何变量,实现多个if else类似的条件分支功能
  • 数组

    • 长度固定
var a [5]int
b := [5]int{1, 2, 3, 4, 5}
a[4] = 100
  • 切片slice
    • 可变长度的数组
    • 可以使用append追加元素,必须把append的结果赋值给原数组,因为如果执行append操作时容量不够,会进行扩容并且返回新的切片
s := make([]string, 3)
s[0] = "a"
s = append(s, "d")
fmt.Println(s[2:5]) // [c d e]
  • map
m := make(map[string]int)
m["one"] = 1
r, ok := m["unknow"]
fmt.Println(r, ok) // 0 false
delete(m, "one")
m2 := map[string]int{"one": 1, "two": 2}
  • range遍历
    • 切片或者map可以使用range来快速遍历
    • 遍历数组时会返回两个值,一个是索引,第二个是对应位置的值,如果不需要索引,可以用下划线来忽略
nums := []int{2, 3, 4}
sum := 0
for i, num := range nums {
   sum += num
}
m := map[string]string{"a": "A", "b": "B"}
for k, v := range m {
   fmt.Println(k, v) // b 8; a A
}
  • 函数
    • 变量类型后置
    • 支持返回多个值
func exists(m map[string]string, k string) (v string, ok bool) {
   v, ok = m[k]
   return v, ok
}
  • 指针
    • 对常用参数进行修改
func add2ptr(n *int) {
   *n += 2
}
  • 结构体
    • 结构体指针避免拷贝等开销且可以修改结构体
type user struct {
   name     string
   password string
}
func main() {
   a := user{name: "wang", password: "1024"}
   var d user
   d.name = "wang"
}
  • 结构体方法
    • 有点像java的成员方法,可以通过结构体去调用这些方法
    • 实现结构体的方法也有两种写法,带指针的能够对结构体进行修改
func (u user) checkPassword(password string) bool {
   return u.password == password
}
func (u *user) resetPassword(password string) {
   u.password = password
}
  • 错误处理
    • 在返回值类型里添加error
    • 对于函数返回值的error,需要判断是否存在,若有 可能需要另行处理
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")
}
  • 字符串
    • 常用的字符串工具函数
    • len函数判断中文字符串的长度可能不是实际字符数量
b := "你好"
fmt.Println(len(b)) // 6
  • 字符串格式化
    • %v代表任意变量,%+v %#v打印出更加详细的信息
s := "hello"
n := 123
fmt.Println(s, n) // hello 123
fmt.Printf("s=%v\n", s)  // s=hello
f := 3.141592653
fmt.Println(f)          // 3.141592653
fmt.Printf("%.2f\n", f) // 3.14
  • JSON处理
    • json.Marshal序列化成字节数组
    • 默认转成字符串输出时是大写的,如果需要小写,在结构体内用json tag语法修改输出的字段名
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)
   }
   //用MarshalIndent返回的字符串有格式,比如每个字段换行、有缩进,可以设置前缀等
   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
   //因为要修改b的值,所以传入指针
   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"}}
}
  • 时间处理
    • 获取当前时间
    • 获取指定时间
    • 获取时间的指定内容如年份、月份
    • 格式化时间,格式化用的是一个特定时间,不是那种yyyy-mm-dd
    • 时间差
    • 从字符串转成时间格式
    • 获取时间戳
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
}
  • 字符串和数字的转换
    • parseInt等函数可以用来解析一个字符串,参数:字符串、进制(0表示自动推测)、位数
    • 用Atoi把一个十进制字符串转成数字,用itoA把数字转成字符串
func main() {
   f, _ := strconv.ParseFloat("1.234", 64)
   fmt.Println(f) // 1.234
   n, _ := strconv.ParseInt("111", 10, 64)
   fmt.Println(n) // 111
}
  • 进程信息
    • os.Args得到程序执行时指定的命令行参数
    • getenv读取环境变量
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
}

标准库

实战

猜谜游戏

  • 生成随机数:使用时间戳作为随机数种子,不设置种子的话会每次生成同样的随机数
  • 读取用户输入:程序在执行时会打开一些文件stdin stdout,可以用NewReader读入使其成为一个reader变量,然后有很多可以操作流的函数。每次读取一行,需要去掉换行符
  • fmt.Scanf可以简化用户输入的读取

命令行词典

  • 调用第三方接口 彩云在线翻译
  • 可以看到header内容很多,自己写请求的话很麻烦,可以用代码自动生成
    • 在浏览器的开发者工具Networks复制想要调用的API(copy as cURL),应该返回一大串JSON
    • 进入代码生成器粘贴刚才的cURL,就会生成代码
  • 发送HTTP请求、处理返回数据
  • 为了避免资源泄露,加一个defer来手动关闭流,会在函数运行结束时执行
  • 请求返回的JSON内容结构很复杂,我们想要把它反序列到一个结构体,可以用代码生成器由JSON自动生成结构体
  • 代码流程:待翻译单词作为参数传入--设置请求结构体--设置请求头--发出请求--读取返回数据--输出指定数据

SOCKS5代理

  • socks5是一种代理协议,明文传输,如某单位的内网有很严格的防火墙策略,如果外网的人需要访问某些资源很麻烦,可以做个socks5代理服务器,相当于在防火墙上开了个口子,让授权的用户可以通过单个端口去访问内部的所有资源
  • 在命令行测试代理服务器:curl -socks5+代理服务器地址+可访问的URL,如果代理服务器正常工作,则会返回
  • socks5的工作原理 image.png
    • 首先是浏览器和代理服务器建立TCP连接,代理再和真正的服务器建立TCP连接,可以分为四个阶段:握手阶段、认证阶段、请求阶段、relay阶段
    • 握手阶段,用户向代理发送请求,包括协议版本号、支持认证的种类,代理服务器会选择其中一个认证方式返回给浏览器,如果返回的是00表示不需要认证,返回其他类型就会开始认证流程,我们实现的是跳过认证流程,实现不加密的方式
    • 请求阶段,浏览器会想代理服务器发出请求,包括版本号、请求的类型,一般主要是connection请求,代表dialing服务器要和某个IP某个端口建立连接。代理服务器收到响应后,会和真正的后端服务器建立连接,然后给浏览器返回一个响应
    • Relay阶段,浏览器会发送正常请求,代理服务器接收到请求后,把请求转换到真正的服务器上,然后转发响应
  • 第一步:实现一个收到什么响应什么的服务器
    • 使用nc命令进行测试,nc 127.0.0.1 1080,输入hello,此时服务器就会返回hello
  • 第二步:实现认证阶段,代理服务器收到浏览器发送的握手阶段的请求的处理过程
  • 第三步:请求阶段,接收并返回浏览器发出的请求
  • 第四步:relay阶段,实现双向数据转发

个人总结:学习了Go的基础语法,并通过三个实战进一步学习Go语言的实际应用。我之前有用过Go,但只是囫囵吞枣,很多都是任务式地复制粘贴推测完成的,通过本次课程结合之前的经验,对Go语言的基础语法有了较为成体系的认知。同时,这三个实战不仅是Go语言的应用,里面还用到很多内容,比如各种标准库的使用、一些高级语法等,我也从中学习到了很多网络相关的知识,仍需要好好消化理解。