Go入门-基础语法(2)| 青训营

71 阅读5分钟

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

入门

基础语法

指针

 package main
 ​
 import "fmt"
 ​
 func add2(n int) {
     n += 2
 }
 ​
 func add2ptr(n *int) {
     *n += 2
 }
 ​
 func main() {
     n := 5
     add2(n)
     fmt.Println(n) // 5
     add2ptr(&n)
     fmt.Println(n) // 7
 }

Go语言也支持指针,但Go语言的指针支持的操作很有限,一个主要用途就是对传入的参数进行修改。

这个add函数试图把一个变量+2,但是单纯像上面这种写法是无效的,因为传入函数的参数实际上是一个拷贝。那也说这个+2是对那个拷贝进行了+2,实际上并不起作用。如果需要起作用的话,那么我们需要把那个类型写成指针类型。为了类型匹配,调用的时候要加一个&符号。

结构体

 package main
 ​
 import "fmt"
 ​
 type user struct {
    name     string
    password string
 }
 ​
 func main() {
    a := user{name: "wang", password: "1024"}
    b := user{"wang", "1024"}
    c := user{name: "wang"}
    c.password = "1024"
    var d user
    d.name = "wang"
    d.password = "1024"
 ​
    fmt.Println(a, b, c, d)                 // {wang 1024} {wang 1024} {wang 1024} {wang 1024}
    fmt.Println(checkPassword(a, "haha"))   // false
    fmt.Println(checkPassword2(&a, "haha")) // false
 }
 ​
 func checkPassword(u user, password string) bool {
    return u.password == password
 }
 ​
 func checkPassword2(u *user, password string) bool {
    return u.password == password
 }

结构体是带类型的字段的集合。

这里的user结构包含了两个字段name和password,我们可以用结构体的名称去初始化一个结构体变量,构造的时候需要传入每个字段的初始值,也可以用键值对的方式去指定初始值,这样可以只对一部分字段进行初始化。结构体也支持指针,这种实现可以避免一些大结构体的拷贝开销。

结构体方法

 package main
 ​
 import "fmt"
 ​
 type user struct {
    name     string
    password string
 }
 ​
 func (u user) checkPassword(password string) bool {
    return u.password == password
 }
 ​
 func (u *user) resetPassword(password string) {
    u.password = password
 }
 ​
 func main() {
    a := user{name: "wang", password: "1024"}
    a.resetPassword("2048")
    fmt.Println(a.checkPassword("2048")) // true
 }

在Go语言中可以为结构体去定义一些方法,类似于其他语言里面的类成员函数。

错误处理

 package main
 ​
 import (
    "errors"
    "fmt"
 )
 ​
 type user struct {
    name     string
    password string
 }
 ​
 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)
    }
 }

在Go语言中,常用的做法是使用了一个单独的返回值来传递错误信息。我们可以用简单的if else语句来处理错误。

在函数里,我们可以在返回值类型里面加一个error,就表示这个函数可能会返回错误。

在函数实现的时候,return需要同时return两个值,如果出现错误的话,那么会return nil和一个error,如果没有错误的话,那么就返回原本的结果和nil。

字符串操作

 package main
 ​
 import (
    "fmt"
    "strings"
 )
 ​
 func main() {
    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
 }

在Go标准库的strings包中有很多常用的字符串工具函数比如contains判断一个字符串里面是否有包含另外一个字符串,count字符串计数,index查找某个字符串的位置,join连接多个字符串,repeat重复多个字符串,replace替换字符串。

字符串格式化

 package main
 ​
 import "fmt"
 ​
 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
 }

在Go标准库的fmt包里有很多字符串格式相关的方法,比如printf类似于C语言中的printf函数,不同点在于Go语言中可以用%v来打印任何类型的变量,不需要区分数字字符串。

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

对于一个已有的结构体,我们可以什么都不做,只要保证每个字段的每一个字母是大写即公开字段,那么这个结构体就能用JSON.marshaler去序列化,变成一个json的字符串。序列化之后字符上也可以用JSON.unmarshaler去反序列化到一个空的变量里面。

时间处理

 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
 }

Go语言中最常用的就是time.now()获取当前时间,也可以用time.date()去构造一个带时区的时间,也可以用sub方法进行时间相减。

数字解析

package main

import (
   "fmt"
   "strconv"
)

func main() {
   f, _ := strconv.ParseFloat("1.234", 64)
   fmt.Println(f) // 1.234

   n, _ := strconv.ParseInt("111", 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
}

Go语言中,字符串和数字类型的转换都在strconv包里。

我们可以用ParseInt或ParseFloat来解析一个字符串。

我们也可以用Atoi将十进制字符串转成数字,用itoA将数字转成字符串。

进程信息

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
}