Golang入门简介与基本语法学习_golang csdn(1),2024年最新我浪费了自己职业生涯的黄金5年

先自我介绍一下,小编浙江大学毕业,去过华为、字节跳动等大厂,目前阿里P7

深知大多数程序员,想要提升技能,往往是自己摸索成长,但自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年最新Golang全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友。
img
img
img
img
img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上Go语言开发知识点,真正体系化!

由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新

如果你需要这些资料,可以添加V获取:vip1024b (备注go)
img

正文

Go语言中的所有参数传递都是值传递。这意味着在调用函数时,实际上是传递了参数值的一个副本。如果想要在函数内部改变参数的值,必须使用指针。

来来来,接着往下看,看看用指针怎么偷梁换柱,把两个数字给掉个儿:

package main

import “fmt”

// 定义一个交换两个数的函数,这里使用指针来直接修改变量的值
func swap(x *int, y *int) {
temp := *x
*x = *y
*y = temp
}

func main() {
a := 20
b := 30
fmt.Println(“交换前 a 和 b 的值:”, a, b) // 交换前 a 和 b 的值: 20 30
swap(&a, &b) // 这里传递的是变量的地址,也就是指向这些变量的指针
fmt.Println(“交换后 a 和 b 的值:”, a, b) // 交换后 a 和 b 的值: 30 20
}

我们通过传递变量地址给swap函数,然后在swap函数内部通过指针去改变这些变量的值,这样改动就能在函数外部看到了。这种方式在Go中获取和修改外部变量的常用手法,就是典型的引用传递。

返回值和命名返回值

Go函数不仅可以返回单个值,还可以返回多个值。当你想从函数返回更多的信息时,这个特性就显得非常有用了。此外,Go还支持命名返回值,也就是在函数声明返回值时给它们命名,这样可以增加代码的清晰度。

package main

import “fmt”

// 定义一个分割整数的函数,返回商和余数
func divmod(a, b int) (int, int) {
quo := a / b // 商
rem := a % b // 余数
return quo, rem
}

// 使用命名返回值的方式来返回多个值
func divmodNamed(a, b int) (quo, rem int) {
quo = a / b // 商
rem = a % b // 余数
return // 不需要明确指定返回值,因为已经在函数签名中命名
}

func main() {
quo, rem := divmod(7, 3)
fmt.Println(“分割7和3得到:商 =”, quo, “余数 =”, rem) // 分割7和3得到:商 = 2 余数 = 1

quoNamed, remNamed := divmodNamed(7, 3)
fmt.Println(“命名返回值分割7和3得到:商 =”, quoNamed, “余数 =”, remNamed) // 命名返回值分割7和3得到:商 = 2 余数 = 1
}

divmodNamed这个函数里,我们就用到了命名返回值。它在返回的时候,就不用再写返回值的名字了,直接一个return搞定,简洁清晰。

复合类型

这些类型可以把简单的类型组合成复杂的数据结构,主要包括数组、切片、映射(map)、结构体(struct)以及它们的方法。

数组和切片

数组是一个固定长度的序列,而切片则是一个可以动态改变大小的序列。切片比数组更常用,因为它们更灵活。

package main

import “fmt”

func main() {
// 数组的定义和初始化
var arr [5]int
arr[0] = 1
arr[1] = 2
// 省略的部分为零值

// 切片的定义和初始化
slice := []int{1, 2, 3, 4, 5} // 切片字面量

// 切片操作
fmt.Println(slice[1:3]) // 输出切片的第2个到第4个元素,不包括索引为3的元素
}

映射(map)

映射是一种无序的键值对的集合。Map是使用哈希表实现的。

package main

import “fmt”

func main() {
// 映射的定义和初始化
m := make(map[string]int)
m[“k1”] = 7
m[“k2”] = 13

// 访问映射
fmt.Println(“map:”, m)

// 删除操作
delete(m, “k2”)
fmt.Println(“map:”, m)
}

结构体(struct)和方法

结构体是一种聚合数据类型,它用于将不同或相同类型的数据组织成一个有意义的单元。

package main

import “fmt”

// 定义结构体
type person struct {
name string
age int
}

// 定义结构体的方法
func (p person) sayHello() {
fmt.Printf(“Hi, I’m %s, %d years old.\n”, p.name, p.age)
}

func main() {
// 初始化结构体
p := person{name: “Jack”, age: 23}

// 调用方法
p.sayHello() // 输出: Hi, I’m Jack, 23 years old.
}

结构体的方法就是那些能够使用该结构体类型的变量或者实例来调用的函数。方法的声明和普通函数类似,只是在方法名前面增加了一个额外的参数,这个参数叫做接收器(receiver),它的类型就是结构体类型。

这些复合类型是Go数据组织的基石,特别是切片和映射,它们在实际开发中运用广泛,可以灵活高效地处理数据集合。结构体通过方法为Go语言提供了面向对象的能力,使得Go能够以更加结构化的方式来处理复杂的数据。

接口

接口在Go中扮演着超级重要的角色,它们让我们的代码更加灵活和模块化。

接口的定义
接口定义了一套方法签名,任何具有这些方法的类型都可以说实现了该接口。

package main

import “fmt”

// 定义一个接口
type Greeter interface {
greet() string
}

// 实现接口的具体类型
type English struct{}

func (e English) greet() string {
return “Hello!”
}

// 实现接口的具体类型
type Chinese struct{}

func (c Chinese) greet() string {
return “你好!”
}

func greetSomeone(g Greeter) {
fmt.Println(g.greet())
}

func main() {
var e English
var c Chinese

greetSomeone(e) // 输出:Hello!
greetSomeone© // 输出:你好!
}

转换和实现

一个类型可以实现多个接口,而一个接口也可以被多个类型实现。类型转换可以在不同的接口和类型之间进行。

package main

import “fmt”

type Walker interface {
walk() string
}

type Talker interface {
talk() string
}

type Human struct{}

func (h Human) walk() string {
return “I’m walking.”
}

func (h Human) talk() string {
return “I’m talking.”
}

func main() {
var h Human

// Human实现了Walker接口
var w Walker = h
fmt.Println(w.walk())

// Human实现了Talker接口
var t Talker = h
fmt.Println(t.talk())
}

接口的组合

接口可以组合其他接口,这样可以很容易地创建出拥有多个方法的新接口。

package main

import “fmt”

type Mover interface {
move() string
}

type Shaker interface {
shake() string
}

// Combiner接口组合了Mover和Shaker接口
type Combiner interface {
Mover
Shaker
}

type Animal struct{}

func (a Animal) move() string {
return “Animal is moving.”
}

func (a Animal) shake() string {
return “Animal is shaking.”
}

func main() {
var a Animal

// Animal实现了Combiner接口
var c Combiner = a
fmt.Println(c.move())
fmt.Println(c.shake())
}

并发编程

Go语言在并发编程方面有着天然的优势,它的设计使得并发变得简单易用。

Go协程(goroutines)

Go协程是轻量级的线程,由Go运行时管理。

package main

import (
“fmt”
“time”
)

func say(s string) {
for i := 0; i < 5; i++ {
time.Sleep(100 * time.Millisecond)
fmt.Println(s)
}
}

func main() {
// 启动一个新的协程
go say(“world”)
// 主协程
say(“hello”)
}

通道(channels)

通道是Go语言中的一种类型,专门用来传递数据,以实现不同的协程之间的通信。

package main

import “fmt”

func main() {
// 创建一个通道
messages := make(chan string)

// 启动一个新的协程
go func() { messages <- “ping” }()

// 从通道接收数据
msg := <-messages
fmt.Println(msg)
}

sync包的使用

sync包提供了基本的同步原语,如互斥锁。

package main

import (
“fmt”
“sync”
)

var wg sync.WaitGroup // 用于等待一组协程完成
var counter int

func main() {
const grs = 2

var mu sync.Mutex // 互斥锁保护计数器

for i := 0; i < grs; i++ {
wg.Add(1)
go func() {
mu.Lock() // 在修改变量之前锁定
{
counter++
}
mu.Unlock() // 修改后解锁
wg.Done()
}()
}

wg.Wait()
fmt.Printf(“Final Counter: %d\n”, counter)
}

错误处理和测试

Go有其独特的错误处理机制。

错误处理机制

在Go中,错误被看作是一种可以预期的结果,而不是异常。

package main

import (
“errors”
“fmt”
)

// 错误处理
func doSomething(flag bool) error {
if !flag {
// 返回一个错误
return errors.New(“something went wrong”)
}
return nil
}

func main() {
err := doSomething(false)
if err != nil {
fmt.Println(err)
}
}

defer、panic和recover

defer用于确保调用在函数结束时发生,panic用于处理不可恢复的错误,recover用于从panic或错误情况中恢复。

package main

import “fmt”

func mayPanic() {
panic(“a problem”)
}

func main() {
defer func() {
if r := recover(); r != nil {
fmt.Println(“Recovered from”, r)
}
}()

mayPanic()

fmt.Println(“After mayPanic()”)
}

编写测试用例

Go的测试框架通过编写_test.go文件来实现。

// 文件名:something_test.go
package main

import “testing”

func TestDoSomething(t *testing.T) {
if err := doSomething(true); err != nil {
t.Errorf(“doSomething(true) should not return error”)
}

if err := doSomething(false); err == nil {
t.Errorf(“doSomething(false) should return error”)
}
}

推荐几个 Golang 学习的文章

Go项目实践

在实际的项目开发中,有一些最佳实践需要遵守。

项目结构规范

遵守一定的目录结构和命名规范,比如使用cmd/pkg/目录来区分命令行应用和库代码。

依赖管理

使用go mod来管理项目依赖。

go mod init github.com/my/repo
go build
go test

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Go)
img

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
05、Go 语言基础语法

Go项目实践

在实际的项目开发中,有一些最佳实践需要遵守。

项目结构规范

遵守一定的目录结构和命名规范,比如使用cmd/pkg/目录来区分命令行应用和库代码。

依赖管理

使用go mod来管理项目依赖。

go mod init github.com/my/repo
go build
go test

网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。

需要这份系统化的资料的朋友,可以添加V获取:vip1024b (备注Go)
[外链图片转存中…(img-WMOfMCTY-1713310022638)]

一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!

  • 9
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值