zl程序教程

您现在的位置是:首页 >  其他

当前栏目

GO语言开篇-Go语言急速入门(基础知识点)| 青训营笔记

2023-02-26 10:15:57 时间

文章目录

一.GO语言应用场景

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

1.1 前言

在学习GO语言的过程中会出一系类的go由浅入深的系列文章,敬请期待!

1.2 什么是GO语言

Go语言是一门现代编程语言,规则简单,统一,优雅,吸收了若干编程语言的优点,解决了C,C++, Python等语言一些固有的难点问题 Go是一种过程编程语言,可用于快速机器代码编译。它是一种静态类型的编译语言。它提供了并发机制,可以轻松开发多核和联网的机器级程序。它是快速,动态类型和解释语言;它提供对接口和类型嵌入的支持。

Go语言是由Google的Robert Griesemer,Rob Pike和Ken Thompson 于2007年开发,但于2009年作为开源编程语言推出。

注: go语言源代码文件的扩展名必须是.go 。

GO语言的优点:

1高性能、高并发 2语法简单、学习曲线平缓 3丰富的标准库 Go具有强大的标准库,以包的形式分发。 4完善的工具链 5静态链接 Go是静态类型语言。因此,在这个编译器中,不仅可以成功编译代码,还可以确保类型转换和兼容性。由于这个特性,Go避免了我们在动态类型语言中遇到的所有问题。 6快速编译 7跨平台 Go语言就像Java语言一样,支持平台独立。由于其模块化设计和模块化,即代码被编译并转换为尽可能小的二进制形式,因此,它不需要依赖性。它的代码可以在任何平台上编译,也可以在任何服务器和应用程序上编译。 8垃圾回收

Go语言的设计者有意识地保持语言简单易懂。整个细节都在少量(一部分)页面中,并且通过语言中的面向对象支持做出了一些有趣的设计决策。对此,语言是固执的,并推荐一种实现事物的惯用方法。它更喜欢组合而不是继承。在Go语言中,“少花钱多办事”就是口头禅。

一个有开发经验的程序员,仅仅耗费一周就可以由学习阶段转到真正的开发阶段,完成一个高并发的应用开发。

1.3 仅仅10行完成高并发的服务器

如下图

哪些公司在使用Go语言:

二.入门

使用Goland,学生可以申请教育免费版:

2.1 基础语法-HelloWorld

Hello World:

package main
​
import (
   "fmt"
)
​
func main() {
   fmt.Println("hello world,上进小菜猪")
}

上述代码解释:

package main:程序的入口包。入口文件

import 的fmt控制输入输出文件:

import ( “fmt” )

main函数调用了fmt的Println

运行结果:

2.2 基础语法-变量类型

2.2.1 变量

声明变量:

1.使用var。

var a = "initial"
var b, c int = 1, 2
var d = true

2.使用变量名:=值。

f := float32(e)
g := a + "foo"

GO会更具上下文的类型来自动确定类型。

3.例子

引用:fmt和math

import (
   "fmt"
   "math"
)

main函数:

var a = "initial"
var b, c int = 1, 2
var d = true
var e float64
f := float32(e)
g := a + "foo"
fmt.Println(a, b, c, d, e, f) // initial 1 2 true 0 0
fmt.Println(g)                // initialapple
const s string = "constant"
const h = 500000000
const i = 3e20 / h
fmt.Println(s, h, i, math.Sin(h), math.Sin(i))

运行结果如下:

2.2.2 if else

1.if后面不需要写括号的:例如

if 7%2 == 0 {
   fmt.Println("7 is even")
} else {
   fmt.Println("7 is odd")
}

2.import的简单写法:

import "fmt"

3.main函数:

if 7%2 == 0 {
   fmt.Println("7 is even")
} else {
   fmt.Println("7 is odd")
}
​
if 8%4 == 0 {
   fmt.Println("8 is divisible by 4")
}
​
if num := 9; num < 0 {
   fmt.Println(num, "is negative")
} else if num < 10 {
   fmt.Println(num, "has 1 digit")
} else {
   fmt.Println(num, "has multiple digits")
}

运行结果如下:

2.2.3 循环

go里面只有for循环,没有while循环,没有do while循环。

简单的for循环如下:

for {
   fmt.Println("loop")
   break
}

可以使用经典的c语言里的for循环:

for j := 7; j < 9; j++ {
   fmt.Println(j)
}

也可以使用continue跳出循环:

for n := 0; n < 5; n++ {
   if n%2 == 0 {
      continue
   }
   fmt.Println(n)
}

运行结果如下:

2.2.4 switch

分支结构和c、c++类似。

不同点:

c/c++如果没有break还会跑下面的同级分支,go则不会。

例子:

1.引入time时间库:

import (
   "fmt"
   "time"
)

2.源码:

先给a赋值为2,然后循环下面的值,运行fmt.Println(“two”),没有break语句也会跳出这个switch。

t := time.Now()使用time库,将现在的系统时间赋值给t,然后进行case条件分支,否则抛出It’s after noon。

a := 2
switch a {
case 1:
   fmt.Println("one")
case 2:
   fmt.Println("two")
case 3:
   fmt.Println("three")
case 4, 5:
   fmt.Println("four or five")
default:
   fmt.Println("other")
}
​
t := time.Now()
switch {
case t.Hour() < 12:
   fmt.Println("It's before noon")
default:
   fmt.Println("It's after noon")
}

运行结果如下:

2.2.5 数组

先定义一个5个长度的数组,然后对第三个位置进行一个赋值,为100.

然后输出a[2]和a的数组长度。

var a [5]int
a[2] = 100
fmt.Println("get:", a[2])
fmt.Println("len:", len(a))

运行结果如下:

定义数组的时候就给其赋值。然后输出b数组看一眼:

b := [5]int{1, 2, 3, 4, 5}
fmt.Println(b)

输出效果如下:

二维数组:定义一个二维数组。使用双层循环对其赋值。

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)

输出结果如下:

2.2.6 切片

定义一个数组,指定其类型为字符。3个空间大小。然后对其进行赋值,然后输出第3个位置,和数组长度,代码如下:

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

代码运行结果如下:

数组追加,和pyrhon一样,使用append,但是注意,语法略微不同。

s = append(s, "d")
s = append(s, "e", "f")
fmt.Println(s) // [a b c d e f]

输出结果如下:

数组的拷贝,使用copt函数,我们先创建一个空的c数组,申请一个大小。然后使用copy函数,将s数组拷贝到c数组。然后输出c数组查看一下效果:

c := make([]string, len(s))
copy(c, s)
fmt.Println(c) // [a b c d e f]

输出结果如下:

切片:s[2:5]:第3个元素开始,到第5个之前(不包括第五个元素)

[:5]:开头到到第5个之前(不包括第五个元素)

s[2:]:第3个元素开始,到最后一个元素。代码如下:

fmt.Println(s[2:5]) // [c d e]
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]

输出结果如下:

2.2.7 map

创建一个map。使用make函数,键值对:string:int。对其进行赋值。

然后输出查看效果,打印长度。指定键,查看值。如下代码:

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

输出结果如下:

加入ok来看看是否有键存在。如下代码:

r, ok := m["unknow"]
fmt.Println(r, ok) // 0 false

输出结果如下:

删除键值对。使用delete函数,代码如下。

delete(m, "one")

赋值案例:

m2 := map[string]int{"one": 1, "two": 2}
var m3 = map[string]int{"one": 1, "two": 2}
fmt.Println(m2, m3)

输出结果如下:

2.2.8 range

for 循环的 range 格式可以对 slice、map、数组、字符串等进行迭代循环。

首先定义一个numw数组,对其赋值。定义sum用于统计累加和。使用for 循环的 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

输出结果如下:

for 循环的 range 格式对map的遍历案例如下:

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
}

输出结果如下:

2.2.9 函数

go语言的函数,有一些区别。变量名在后面。

先看一下主函数:

res := add(1, 2)
fmt.Println(res) // 3

然后看一下add函数,这个add函数的作用是返回俩个值相加的结果。

func add(a int, b int) int {
   return a + b
}

然后我们看一下返回多值的情况:

传入俩个参数,一个map,一个字符a。

v, ok := exists(map[string]string{"a": "A"}, "a")
fmt.Println(v, ok) // A True

看一下exists的函数。这里接受俩个函数,m[“a”]存在,ok的值为ture。返回给主函数。

func exists(m map[string]string, k string) (v string, ok bool) {
   v, ok = m[k]
   return v, ok
}

输出结果如下:

2.2.10 指针

go语言的指针功能非常有限,主要的功能是为了对传入的参数进行修改。

这里的传入的参数实际上是一个拷贝。拷贝+1是不起作用的。

func add2(n int) {
   n += 2
}

想要起作用只能采用指针类型:

func add2ptr(n *int) {
   *n += 2
}

为了类型匹配,调用的时候,需要加&:add2ptr(&n)

下面是main函数:

n := 5
add2(n)
fmt.Println(n) // 5
add2ptr(&n)
fmt.Println(n) // 7

运行结果如下:

2.2.11 结构体

先定义一个结构体user,定义两个变量:name和password:

type user struct {
   name     string
   password string
}

定义结构体值的几个方法:

下面代码是4种对结构体类型的赋值方法,非常简单。如下:

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}

运行结果如下:

使用函数:

普通函数(不使用指针。)

func checkPassword(u user, password string) bool {
   return u.password == password
}

使用指针:

func checkPassword2(u *user, password string) bool {
   return u.password == password
}

主函数,调用:

fmt.Println(checkPassword(a, "haha"))   // false
fmt.Println(checkPassword2(&a, "haha")) // false

运行结果如下:

2.2.12 结构体方法

这里先把a赋值一个user结构体类型,并且对其进行赋值。

a := user{name: "wang", password: "1024"}
a.resetPassword("2048")

然后直接a.调用方法:

使用指针才能改变其结构体里的值。

func (u *user) resetPassword(password string) {
   u.password = password
}

检查一下结构体里的值是不是等于2048。

fmt.Println(a.checkPassword("2048")) // true

checkPassword函数的内容如下:

func (u user) checkPassword(password string) bool {
   return u.password == password
}

运行结果如下:

2.2.13 错误处理

导入依赖包:errors,代码如下:

import (
   "errors"
   "fmt"
)

先看一下调用的findUser,我们先传入了应该数组,和一个字符串。

u, err := findUser([]user{{"wang", "1024"}}, "wang")

findUser如下:

这个函数作用是检查结构体的值,是不是我们规定的难搞值,如果是的话返回return &u, nil。否则就是出错了。返回return nil, errors.New(“not found”),将not found 写到日志里,代码如下:

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

在主函数里,如果err不是nil的话,说明出错了,我们输出错误日志看一眼。

if err != nil {
   fmt.Println(err)
   return
}

输出u的name

fmt.Println(u.name) // wang

来一个错误案例:

我们调用findUser,但是传入 “li”,上面的代码当然会u.name != name,就会有日志存入,err,然后下面进入判断,输出err,代码如下:

if u, err := findUser([]user{{"wang", "1024"}}, "li"); err != nil {
   fmt.Println(err) // not found
   return
} else {
   fmt.Println(u.name)
}

运行结果如下:

2.2.14 字符串操作

导入依赖包:strings,代码如下:

import (
   "fmt"
   "strings"
)

方法列表:

主函数代码,对字符串操作案例:

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

运行结果如下:

2.2.15 字符串格式化

先分别定义一个字符串,整数,指针。分别打印一下,代码如下:

s := "hello"
n := 123
p := point{1, 2}
fmt.Println(s, n) // hello 123
fmt.Println(p)    // {1 2}

运行结果如下:

可以使用任意%v来输出任意的的数值。代码如下:

可以使用%+v来输出详细的数值。

可以使用%#v来输出更加详细的数值。

\n标识换行。

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}

运行结果如下:

小数,规定位数,这里和c/c++一模一样。

f := 3.141592653
fmt.Println(f)          // 3.141592653
fmt.Printf("%.2f\n", f) // 3.14

2.2.16 json处理

导入依赖包: “encoding/json”,代码如下:

import (
   "encoding/json"
   "fmt"
)

定义一个结构体userInfo。

type userInfo struct {
   Name  string
   Age   int `json:"age"`
   Hobby []string
}

对其进行一个赋值操作。

a := userInfo{Name: "wang", Age: 18, Hobby: []string{"Golang", "TypeScript"}}

使用json的Marshal方法。我们先看一下Marshal方法:

func Marshal(v interface{}) ([]byte, error)

从返回值我们可以看到,该函数有两个返回值,一个是传入参数v的json编码,类型为[]byte,另外一个就是error。

将结构体变量a转换为json,代码如下:

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

运行结果如下:

使用MarshalIndent:

func MarshalIndent(v any, prefix, indent string)([]byte, error)

MarshalIndent 类似于 Marshal,但应用缩进来格式化输出。根据缩进嵌套,输出中的每个 JSON 元素都将在以前缀开头的新行开始,后跟一个或多个缩进副本。

buf, err = json.MarshalIndent(a, "", "\t")
if err != nil {
   panic(err)
}
fmt.Println(string(buf))

输出如下:

JSON解码函数Unmarshal:

func Unmarshal(data []byte, v interface{}) error

Unmarshal函数解析json编码的数据并将结果存入v指向的值。

Unmarshal和Marshal做相反的操作,必要时申请映射、切片或指针,有如下的附加规则:

要将json数据解码写入一个指针,Unmarshal函数首先处理json数据是json字面值null的情况。此时,函数将指针设为nil;否则,函数将json数据解码写入指针指向的值;如果指针本身是nil,函数会先申请一个值并使指针指向它。

要将json数据解码写入一个结构体,函数会匹配输入对象的键和Marshal使用的键(结构体字段名或者它的标签指定的键名),优先选择精确的匹配,但也接受大小写不敏感的匹配。

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

输出结果如下:

2.2.17 时间处理

导入依赖包: “time”,代码如下:

import (
   "fmt"
   "time"
)

获取当前的时间并且输出:

now := time.Now()
fmt.Println(now) 

运行结果如下:

还可以自己构造一个时间,如下代码:

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

我们指定输出格式,代码运行结果截图如下:

时间差:使用sub函数,可以得出俩个时间的时间差。

diff := t2.Sub(t)
fmt.Println(diff)                           // 1h5m0s
fmt.Println(diff.Minutes(), diff.Seconds()) // 65 3900

代码运行结果截图如下:

time.Parse()函数

Go语言中的Parse()函数用于解析格式化的字符串,然后查找它形成的时间值。

官方api:

func Parse(layout, value string) (Time, error)

在这里,layout通过以哪种方式显示参考时间(即定义为Mon Jan 2 15:04:05 -0700 MST 2006)来指定格式,如果它是该值的话。但是,先前定义的布局(例如UnixDate,ANSIC,RFC3339等)解释了标准时间以及参考时间的合适表示形式。并且value参数保存字符串。其中,从值中删除的元素假定为零,如果不可能为零,则假定为一。

返回值: 它返回它表示的时间值。并且如果不存在时区指示符,则它会返回UTC时间。

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

运行结果如下:

2.2.18 数字解析

导入依赖包:“strconv”,代码如下:

import (
   "fmt"
   "strconv"
)

可以使用ParseFloa或者ParseInt转换小数或者整数。

f, _ := strconv.ParseFloat("1.234", 64)
    fmt.Println(f) // 1.234
​
n, _ := strconv.ParseInt("111", 10, 64)
    fmt.Println(n) // 111

上面代码整数的第二个参数代表转换数字的进制,这里是10进制。

如果是0的话就是自动转换。代码如下;

n, _ = strconv.ParseInt("0x1000", 0, 64)
fmt.Println(n) // 4096

使用Atoi快速把字符串转换为数字:

n2, _ := strconv.Atoi("123")
fmt.Println(n2) // 123

数字不合法处理:

n2, err := strconv.Atoi("AAA")
fmt.Println(n2, err) // 0 strconv.Atoi: parsing "AAA": invalid syntax

运行结果如下:

2.2.19 进程信息

导入依赖包:“os"和"os/exec”,代码如下:

import (
   "fmt"
   "os"
   "os/exec"
)

输出命令行参数:

fmt.Println(os.Args)

获取或者写入环境变量:

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

运行结果如下:

三.本文总结

本文写了下面几大块:

  • 什么是 G0语言
  • Go 语言基本特征
  • Go 语言应用优势
  • G0语言入门
  • 开发环境-安装 Golang
  • G0语言基础语法

下篇我将撰写Go语言的实战应用的笔记,敬请期待!