这是我参与「第三届青训营 -后端场」笔记创作活动的的第1篇笔记。
第一个go程序:
package main
import "fmt"
func main() {
fmt.Println("hello go!")
}
第一行:代表该文件属于main包的一部分,main包为程序的入口包;
第二行:导入了标准库中的 fmt包,主要用于向屏幕输入输出字符串;
第三行:func为函数关键字 main()为函数名;
第四行:使用fmt包中的Printf()函数打印字符串。
注意点:import导入的包一定要使用,如果不使用会报错。
一条代码语句最后,没有分号!!
在一行中,只能写一条代码语句,因为Go编译器式一行行编译的。
变量:
变量的声明:
var a = "initial"
b := a + "int";
fmt.Println(b)
注意:声明变量有两种方式;
声明的变量一定要使用。
常量:
const c = "123456";
const d = 12;
const e = true;
常量的关键字为:const,常量没有确定的类型,会根据其上下文自动确定类型。
if:
if d >= 90 {
fmt.Println("优秀!")
}else if d >= 60{
fmt.Println("及格!")
}else{
fmt.Println("不及格!")
}
使用和Java中类似,注意:if后没有括号()
for:
for{
fmt.Println("for循环")
break
}
for i := 0; i < 5; i++{
fmt.Println( i );
}
n := 10
for n > 0{
fmt.Println( n )
n--
}
switch:
m := 2
switch m {
case 1:
fmt.Println("one")
case 2:
fmt.Println("two")
case 3:
fmt.Println("three")
default:
fmt.Println("other")
}
执行结果:two
go中的switch不需要加break。
数组:
func main() {
//声明数组变量,默认全部元素为0
var a [5]int
a[4] = 100
fmt.Println("get:", a[2]) // get: 0
fmt.Println("len:", len(a))// len: 5
b := [5]int{1, 2, 3, 4, 5}
fmt.Println(b) // [1 2 3 4 5]
var twoD [2][3]int
for i := 0; i < 2; i++ {
for j := 0; j < 3; j++ {
twoD[i][j] = i + j
}
}
fmt.Println("2d: ", twoD)
}
切片:
s := make([]string, 3)
s[0] = "a"
s[1] = "b"
s[2] = "c"
fmt.Println("get:", s[2]) // c
fmt.Println("len:", len(s)) // 3
//扩容
s = append(s, "d")
s = append(s, "e", "f")
fmt.Println(s) // [a b c d e f]
c := make([]string, len(s))
copy(c, s)
fmt.Println(c) // [a b c d e f]
//注意:前闭后开
fmt.Println(s[2:5]) // [c d e] 取出索引为2 ~ 5-1 的元素
fmt.Println(s[:5]) // [a b c d e]
fmt.Println(s[2:]) // [c d e f]
good := []string{"g", "o", "o", "d"}
fmt.Println(good) // [g o o d]
注意切片的扩容使用的是:append()函数;
注意切片s[m,n]:取出的是索引为[m,n)的元素,前闭后开!
map(最常用!):
func main() {
m := make(map[string]int)
m["one"] = 1
m["two"] = 2
fmt.Println(m) // map[one:1 two:2]
fmt.Println(len(m)) // 2
fmt.Println(m["one"]) // 1
fmt.Println(m["unknow"]) // 0
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}
fmt.Println(m2, m3)
}
注意:map的遍历,顺序是随机的!!
func() 函数:
//函数的参数列表: 参数名 参数类型
func add(a int, b int) int {
return a + b
}
func add2(a, b int) int {
return a + b
}
//可以设置多个返回值
func exists(m map[string]string, k string) (v string, ok bool) {
v, ok = m[k]
return v, ok
}
func main() {
res := add(1, 2)
fmt.Println(res) // 3
v, ok := exists(map[string]string{"a": "A"}, "a")
fmt.Println(v, ok) // A True
}
range:
func main() {
nums := []int{2, 3, 4}
sum := 0
//使用range遍历数组: 第一个返回值为索引 第二个返回值为元素值
for i, num := range nums {
sum += num
if num == 2 {
fmt.Println("index:", i, "num:", num) // index: 0 num: 2
}
}
fmt.Println(sum) // 9
//使用range遍历map,第一个返回值为key 第二个返回值为value
m := map[string]string{"a": "A", "b": "B"}
for k, v := range m {
fmt.Println(k, v) // b 8; a A
}
//可以单独取出key
for k := range m {
fmt.Println("key", k) // key a; key b
}
//可以单独取出value
for _, v := range m {
fmt.Println("value", v) // value B; value A
}
}
结构体:
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
}
注意定义结构体变量的几种方法;
注意:go中方法的参数列表为 参数名 参数类型
结构体方法:
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
}
比较这里的结构体方法和结构体部分中的方法传入结构体参数:结构体方法就是将结构体放在了方法名之前,而不是将其作为参数传递。
错误处理:
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)
}
}
如果一个函数可能产生错误,那么就把错误作为一个返回值返回;
在调用可能出现错误的函数时,要注意判断错误是否为nil。
字符串操作:
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
json处理:
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"}}
}
注意:结构体中的属性名全部以大写字母开头;
然后就可以使用 buf, err := json.Marshal( a )来序列化一个结构体对象a;
使用json.Unmarshal(buf, &b)进行反序列化,并将值赋给结构体对象b。
时间处理:
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 //获取时间戳
}
数字解析(字符串->数字):
import (
"fmt"
"strconv"//使用strconv这个包进行解析
)
func main() {
f, _ := strconv.ParseFloat("1.234", 64)
fmt.Println(f) // 1.234
n, _ := strconv.ParseInt("111", 10, 64)//10代表十进制 64代表64位精度
fmt.Println(n) // 111
n, _ = strconv.ParseInt("0x1000", 0, 64)//如果第二个参数为0,代表自行推测进制
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
}
进程信息:
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
}