标签 GC 下的文章

Go的“七宗罪”:一篇“Go依然不够好”如何引爆社区激辩?

本文永久链接 – https://tonybai.com/2025/08/25/go-is-still-not-good

大家好,我是Tony Bai。

在技术圈,平静的湖面下往往暗流涌动。对于Go语言社区而言,这股潜藏已久的暗流,被近期的一篇名为《Go is still not good》的博文彻底引爆。作者Thomas Habets,一位自称拥有超过十年Go使用经验的资深开发者,在他的这篇文章中系统性地列举了他眼中Go语言的“七宗罪”。这篇文章迅速登上Hacker News热榜,吸引了超过700条评论,形成了一场规模空前的社区大辩论。

参与者中不乏Go的早期采纳者、贡献者和日常重度使用者。他们争论的焦点,早已超越了语法糖的优劣,直指Go语言最核心的设计哲学——那些曾被誉为“简单”和“务实”的基石,如今在一些开发者眼中,却成了束缚发展、埋下隐患的“原罪”。

在这篇文章中,我就和大家一起跟随这场激辩,逐一剖析这引发轩然大波的“七宗罪”,看看从中能得到哪些有益的启示。

第一宗罪:歧义之空——nil 的双重身份

这是Go语言中最著名的“陷阱”,也是原文作者打响的第一枪。一个持有nil指针的接口变量,其自身并不等于nil。

package main

import "fmt"

type Error interface {
    Error() string
}

type MyError struct{}

func (e *MyError) Error() string { return "my error" }

func GetError() *MyError {
    // 假设在某种条件下,我们返回一个 nil 的具体错误类型指针
    return nil
}

func main() {
    var err Error = GetError()

    // 输出: false
    // 尽管接口 err 内部持有的值是 nil,但接口本身因为包含了类型信息 (*MyError),所以它不为 nil。
    fmt.Println(err == nil) 

    if err != nil {
        // 这段代码会被执行,然后可能在后续操作中引发 panic
        fmt.Printf("An error occurred: %v (type: %T)\n", err, err)
        // err.Error() // 若MyError的Error方法有解引用操作,此处会panic
    }
}

我们知道:Go的接口(interface)在内部实现为一个包含两部分的“胖指针”(fat pointer):一个指向类型元数据的指针和一个指向实际数据的指针。只有当这两个指针都为nil时,接口变量本身才被认为是nil。在上述例子中,err的内部状态是(type=*MyError, value=nil)。因为类型信息存在,err != nil的判断为真,导致程序逻辑错误地进入了错误处理分支,挑战了开发者的常规直觉。

社区激辩

  • 批评者阵营:Hacker News上,有用户提供了一个经典的Playground示例,展示了这个问题如何在生产环境中导致panic,并评论道:“这确实会在生产中咬你一口,而且在代码审查中极易被忽略。”另一位用户则更为尖锐,他引用了Rob Pike关于Go是为“非研究型、刚毕业的年轻工程师”设计的言论,反问道:“一个声称为了简化编程而设计的语言,却包含如此令人困惑的nil行为,这本身就是一种讽刺。”

  • 辩护者阵营:另一派观点认为,这并非缺陷,而是Go底层数据结构逻辑的直接体现。有开发者解释道:“接口值是一个包含类型和值的偶对。(&Cat, nil)当然不等于(nil, nil)。”他们认为,一旦理解了接口的内存模型,这个问题便不再神秘,甚至可以利用这一特性(例如,在nil接收者上调用方法)。然而,这种辩护本身就强化了批评者的观点:一门标榜高级和简单的语言,却要求开发者对底层的实现细节有如此深刻的理解,这是否可以看作设计上的一种失败呢?

第二宗罪:作用域之惑——被迫扩展的err变量生命周期

Go通过if err := foo(); err != nil语法,优雅地将err变量的作用域限制在if块内,这被广泛认为是最佳实践。然而,当函数调用需要返回除error之外的值时,这种优雅便荡然无存。

bar, err := foo()
if err != nil {
    return err
}
// 此处的err变量将在整个函数剩余部分都有效,即使它现在的值是nil

if err = foo2(); err != nil { // 复用err
    return err
}

// ... 大量代码 ...

return err

Go的短变量声明:=要求左侧至少有一个新变量。为了接收bar这个新值,err也被迫在函数作用域内被重新声明(或首次声明)。这导致err的生命周期被人为地拉长,污染了整个函数的作用域。

社区激辩

  • 批评者阵营:原文作者尖锐地指出,这种设计“强迫你做错误的事情”。一个本应是局部的错误变量,现在却像个幽灵一样在整个函数中游荡,增加了代码阅读者的认知负担。读者必须时刻追踪err变量最后一次被赋值的位置,这极易导致bug,尤其是在重构或修改长函数时。
  • 辩护者阵营:对此的辩护声音较弱,大多认为这是个“可以忍受的小麻烦”。他们认为,这是为了保持语法一致性(:=的规则)而付出的代价。然而,这恰恰暴露了Go在追求一种形式上的“简单”时,牺牲了更重要的“上下文清晰性”。

第三宗罪:所有权之乱——append的隐式副作用

slice是Go的基石之一,但其与底层数组(backing array)的模糊关系,通过append函数暴露无遗,构成了另一个经典的“搬起石头砸自己的脚”。

原文的例子一针见血地揭示了append行为的不可预测性:

package main

import "fmt"

func main() {
    // 案例一:当容量足够时,发生“幽灵写入”
    a := []string{"hello", "world", "!"}
    b := a[:1]                 // b与a共享底层数组,且cap(b) == 3
    b = append(b, "NIGHTMARE") // 修改了b,因为容量足够,直接修改了底层数组
    fmt.Println(a)// 结果:a变成了[hello NIGHTMARE !]

    // 案例二:当容量不足时,修改“失败”
    a = []string{"hello", "world", "!"}
    b = a[:1]
    b = append(b, "BACON", "THIS", "SHOULD", "WORK") // 容量不足,分配了新数组
    fmt.Println(a)// 结果:a依然是[hello world !]
}

我们知道:append的行为取决于slice的容量(cap)。如果追加后未超出容量,它会就地修改底层数组;否则,会分配一个新的、更大的数组。这种设计不仅让append的性能变得不确定,更严重的是,它破坏了函数调用的封装性,使得slice既不像值类型(可能被远程修改),也不像纯粹的引用类型(可能因重分配而断开联系)。

社区激辩

  • 批评者阵营:Hacker News上一位获得高赞的评论是这样的:“append的例子是Go缺陷中最恶劣、最不可原谅的。”这种行为使得数据流变得难以追踪,迫使开发者必须时刻警惕slice的容量,或养成防御性编程的习惯,例如总是重新接收append的返回值。这与Go追求的“明确”背道而驰。
  • 辩护者阵营:支持者认为这是为了性能做出的合理权衡,避免了不必要的内存分配。他们强调,Go官方文档已明确说明了slice的工作原理。然而,这再次回到了那个核心问题:一门标榜“简单”的语言,是否应该包含如此微妙且需要深度理解才能安全使用的核心数据结构?

第四宗罪:作用域陷阱——函数级的defer

defer是Go处理资源释放的利器,但它的作用域是整个函数,而非其所在的词法块(lexical scope)。这在循环中处理资源时会成为一个严重的资源泄漏问题。

for _, file := range files {
    f, err := os.Open(file)
    if err != nil { /* ... */ continue }
    // defer不会在每次循环结束时执行,而是堆积到函数返回时执行
    // 如果文件列表很长,将耗尽文件句柄
    defer f.Close()
    // ... process file
}

根本原因在于defer语句的执行被推入一个与当前函数关联的栈中,在函数返回前统一执行。这简化了编译器的实现,并确保了panic时资源也能被释放。

社区激辩

  • 批评者阵营:一个开发者的高赞评论代表了社区的普遍困惑:“我至今不明白defer为什么是函数作用域而非词法作用域。”这与C++的RAII或Java的try-with-resources相比,是一种设计上的倒退。公认的解决方法是使用匿名函数func(){…}()包裹循环体,但这无疑增加了代码的丑陋和复杂性。
  • 辩护者阵营:有用户指出,函数级作用域也有其便利之处,例如可以在if块中有条件地注册一个defer。但总体而言,社区普遍认为,默认应该是更安全、更符合直觉的词法作用域。

第五宗罪:异常之隐——被标准库“吞噬”的panic

Go的哲学是:error用于可预见的错误,panic用于程序无法继续的灾难。然而,作者指出,标准库中的fmt.Print和net/http服务器等关键部分,会主动recover从panic中恢复,这破坏了panic的基本约定。

这意味着开发者必须编写“异常安全”的代码。你必须假设任何传递给标准库的代码都可能在panic后被恢复。因此,像互斥锁(mutex)这样的资源必须通过defer来确保释放,否则一旦发生被“吞噬”的panic,就会造成死锁。作者愤怒地指出:“所有希望都破灭了。你必须写异常安全的代码,但你又不应该使用异常。你只能承受异常带来的所有负面影响。”

社区激辩:这一点在社区中几乎没有辩护的声音。这被视为一种设计上的不一致和“伪善”。语言在表层倡导一种错误处理哲学,却在底层库中悄悄破坏它,迫使开发者为这种矛盾买单。

第六宗罪:编码之殇——对非UTF-8的“绥靖政策”

Go的string类型本质是只读的[]byte,不强制其为合法的UTF-8。这在与操作系统交互(如处理文件名)时提供了灵活性,但也埋下了隐患。

作者控诉,这种“宽松”策略是数据丢失的根源。当工具不假思索地按UTF-8处理文件名时,遇到非UTF-8编码的文件名可能会跳过或处理失败,导致在备份、恢复等关键操作中“静默地”遗漏数据。

社区激辩

  • 批评者阵营:他们认为类型系统应防止此类错误。有用户激烈地评论道:“Go让你很容易做那些看起来99.7%的时间都有效,但却是愚蠢、错误、不正确的事情……然后有一天,你的用户就因为一个非UTF-8文件名而永久丢失了数据。”
  • 辩护者阵营:另一方则认为Go的做法才是务实的。有用户指出,一个强制Unicode正确性的文件接口在真实世界中是有问题的。Rust的OsStr虽然严谨,但人体工程学极差。Go的方式虽然“混乱”,但在实践中更方便。这揭示了严谨性与便利性之间的深刻矛盾。

第七宗罪:承诺之虚——伪善的“简单”与被忽视的性能

这并非单一技术点,而是对Go整体设计理念的综合批判。

  • 简单性的代价是复杂性转移:许多评论者指出,Go语言层面的“简单”,是把复杂性推给开发者来承担。没有枚举、没有强大的泛型(即使1.18加入了,也限制颇多)、没有Result类型,导致开发者需要手写大量重复的样板代码和自定义数据结构。
  • 内存管理的“信任危机”:原文作者提到“RAM is cheap”是危险的思维。Hacker News上有开发者分享了其在内存敏感项目中被Go的非压缩GC和堆碎片化问题折磨的经历,他们甚至不得不重写部分标准库以避免内存分配。这与Go宣称的“高性能”和“无忧GC”形成了鲜明对比。

为何着一篇文章能掀起千层浪?

这场激辩之所以如此激烈,是因为它触及了Go社区内部长期存在的深层张力:

  1. “Google的Go” vs “世界的Go”:Go的许多设计源于解决Google内部特定问题的需求(C++编译慢、monorepo文化)。这种“出身”决定了它在某些方面与更广阔的编程世界存在脱节。早年对单调时钟的忽视就是典型例子。
  2. 简单主义 vs 现代语言特性:Go的创造者们带着一种“回归本源”的复古主义情怀,刻意回避了过去几十年编程语言理论的发展成果,如高级类型系统、代数数据类型等。这使得Go易于上手,但也让它在处理复杂逻辑时显得捉襟见肘,迫使开发者“用代码的冗余换取语言的简单”。
  3. 显式 vs 便利:if err != nil是显式的,但它不便利。Result类型和?操作符是便利的,但它在某种程度上是隐式的。Go坚定地站在了“显式”这一边,但社区中渴望“便利”的声音从未停止。

小结

将Go的这些“罪状”简单归结为“错误”也是片面的。它们是Go强硬的、自洽的设计哲学所带来的必然产物。

  • 这是一门有“历史”的现代语言:Go的设计深受其创造者们在C、Unix、Plan 9上的经验影响。它继承了C的简洁,但也继承了其对底层细节的暴露。
  • 承认权衡,理解其生态位:Go在“开发效率”、“运行性能”和“语言简单性”之间做出了明确的取舍,在云原生、微服务领域找到了无与伦比的“甜蜜点”。
  • 缓慢的进化也是一种承诺:Go团队对语言的改变极为谨慎,以维护其著名的向后兼容性承诺。但它并非一成不变。泛型的加入、for range循环变量作用域的修正,都表明Go在倾听社区的声音。

《Go is still not good》及其引发的激辩,为我们提供了一个宝贵的窗口,去重新审视这门既年轻又充满“历史感”的语言。它提醒我们,没有完美的语言,只有充满权衡的工具。

对于Go开发者而言,理解这“七宗罪”的来龙去脉,不仅能帮助我们写出更健壮、更地道的代码,更能让我们清晰地认识到Go的优势与边界。与其无休止地争论它是否“足够好”,不如深入思考:它是否是解决我们当前问题的正确工具? 而这,或许才是这场大辩论给予我们的最大启示。


你的Go技能,是否也卡在了“熟练”到“精通”的瓶颈期?

  • 想写出更地道、更健壮的Go代码,却总在细节上踩坑?
  • 渴望提升软件设计能力,驾驭复杂Go项目却缺乏章法?
  • 想打造生产级的Go服务,却在工程化实践中屡屡受挫?

继《Go语言第一课》后,我的《Go语言进阶课》终于在极客时间与大家见面了!

我的全新极客时间专栏 《Tony Bai·Go语言进阶课》就是为这样的你量身打造!30+讲硬核内容,带你夯实语法认知,提升设计思维,锻造工程实践能力,更有实战项目串讲。

目标只有一个:助你完成从“Go熟练工”到“Go专家”的蜕变! 现在就加入,让你的Go技能再上一个新台阶!


商务合作方式:撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求,请扫描下方公众号二维码,与我私信联系。

Go 1.25中值得关注的几个变化

本文永久链接 – https://tonybai.com/2025/08/15/some-changes-in-go-1-25

大家好,我是Tony Bai。

北京时间2025年8月13日,Go 团队如期发布了 Go 语言的最新大版本——Go 1.25。按照惯例,每次 Go 大版本发布时,我都会撰写一篇“Go 1.x 中值得关注的几个变化”的文章。自 2014 年的 Go 1.4 版本起,这一系列文章已经伴随大家走过了十一个年头。

不过,随着我在版本冻结前推出的“Go 1.x 新特性前瞻”系列,以及对该大版本可能加入特性的一些独立的解读文章,本系列文章的形式也在不断演变。本文将不再对每个特性进行细致入微的分析,因为这些深度内容大多已在之前的《Go 1.25 新特性前瞻》一文中详细讨论过。本文将更聚焦于提炼核心亮点,并分享一些我的思考。

好了,言归正传,我们来看看Go 1.25带来了哪些惊喜!

语言变化:兼容性基石上的精雕细琢

正如 Go 一贯所做的,新版 Go 1.25 继续遵循 Go1 的兼容性规范。最令 Gopher 们安心的一点是:Go 1.25 没有引入任何影响现有 Go 程序的语言级变更

There are no languages changes that affect Go programs in Go 1.25.

这种对稳定性的极致追求,是 Go 成为生产环境首选语言之一的重要原因。

尽管语法层面波澜不惊,但语言规范内部却进行了一次“大扫除”——移除了“core types”的概念。这一变化虽然对日常编码无直接影响,但它简化了语言规范,为未来泛型可能的演进铺平了道路,体现了 Go 团队在设计层面的严谨与远见。关于此变化的深度解读,可以回顾我之前的文章《Go 1.25 规范大扫除:移除“Core Types”,为更灵活的泛型铺路》。

编译器与运行时:看不见的性能飞跃

如果说 Go 1.24 的运行时核心是优化 map,那么 Go 1.25 的灵魂则在于让 Go 程序更“懂”其运行环境,并对 GC 进行了大刀阔斧的革新。

容器感知型 GOMAXPROCS

这无疑是 Go 1.25 最具影响力的变化之一。在容器化部署已成事实标准的今天,Go 1.25 的运行时终于具备了 cgroup 感知能力。在 Linux 系统上,它会默认根据容器的 CPU limit 来设置 GOMAXPROCS,并能动态适应 limit 的变化。

这意味着,只需升级到 Go 1.25,你的 Go 应用在 K8s 等环境中的 CPU 资源使用将变得更加智能和高效,告别了过去因 GOMAXPROCS 默认值不当而导致的资源浪费或性能瓶颈。更多细节,请参阅我的文章《Go 1.25 新提案:GOMAXPROCS 默认值将迎 Cgroup 感知能力,终结容器性能噩梦?》。

实验性的 Green Tea GC

Go 1.25 迈出了 GC 优化的重要一步,引入了一个新的实验性垃圾收集器。通过设置 GOEXPERIMENT=greenteagc 即可在构建时启用。

A new garbage collector is now available as an experiment. This garbage collector’s design improves the performance of marking and scanning small objects through better locality and CPU scalability.

据官方透露,这个新 GC 有望为真实世界的程序带来 10%—40% 的 GC 开销降低。知名go开发者Josh Baker(@tidwall)在Go 1.25发布正式版后,在X上分享了自己使用go 1.25新gc(绿茶)后的结果,他开源的实时地理空间和地理围栏项目tile38的GC开销下降35%:

这是一个巨大的性能红利,尤其对于重度依赖GC的内存密集型应用。虽然它仍在实验阶段,但其展现的潜力已足够令人兴奋。对 Green Tea GC 设计原理感兴趣的朋友,可以阅读我的文章《Go 新垃圾回收器登场:Green Tea GC 如何通过内存感知显著降低 CPU 开销?》。

此外,Go 1.25 还修复了一个存在于 Go 1.21 至 1.24 版本中可能导致 nil pointer 检查被错误延迟的编译器 bug,并默认启用了 DWARFv5 调试信息,进一步缩小了二进制文件体积并加快了链接速度,对DWARFv5感兴趣的小伙伴儿可以重温一下我之前的《Go 1.25链接器提速、执行文件瘦身:DWARF 5调试信息格式升级终落地》一文,了解详情。

工具链:效率与可靠性的双重提升

强大的工具链是 Go 生产力的核心保障。Go 1.25 在此基础上继续添砖加瓦。

go.mod 新增 ignore 指令

对于大型 Monorepo 项目,go.mod 新增的 ignore 指令是一个福音。它允许你指定 Go 命令在匹配包模式时应忽略的目录,从而在不影响模块依赖的前提下,有效提升大型、混合语言仓库中的构建与扫描效率。关于此特性的详细用法,请见《Go 工具链进化:go.mod 新增 ignore 指令,破解混合项目构建难题》。

支持仓库子目录作为模块根路径

一个长期困扰 Monorepo 管理者和自定义 vanity import 用户的难题在 Go 1.25 中也得到了解决。Go 命令现在支持在解析 go-import meta 标签时,通过新增的 subdir 字段,将 Git 仓库中的子目录指定为模块的根。

这意味着,你可以轻松地将 github.com/my-org/my-repo/foo/bar 目录映射为模块路径 my.domain/bar,而无需复杂的代理或目录结构调整。这个看似微小但备受期待的改进,极大地提升了 Go 模块在复杂项目结构中的灵活性。想了解其来龙去脉和具体配置方法,可以参考我的文章《千呼万唤始出来?Go 1.25解决Git仓库子目录作为模块根路径难题》。

go doc -http:即开即用的本地文档

这是一个虽小但美的改进。新的 go doc -http 选项可以快速启动一个本地文档服务器,并在浏览器中直接打开指定对象的文档。对于习惯于离线工作的开发者来说,这极大地提升了查阅文档的便捷性。详细介绍见《重拾精髓:go doc -http 让离线包文档浏览更便捷》。

go vet 新增分析器

go vet 变得更加智能,新增了两个实用的分析器:

  • waitgroup:检查 sync.WaitGroup.Add 的调用位置是否错误(例如在 goroutine 内部调用)。
  • hostport:诊断不兼容 IPv6 的地址拼接方式 fmt.Sprintf(“%s:%d”, host, port),并建议使用 net.JoinHostPort。

这些静态检查能帮助我们在编码阶段就扼杀掉一批常见的并发和网络编程错误。

标准库:功能毕业与实验探索

标准库的演进是每个 Go 版本的重要看点。

testing/synctest 正式毕业

在 Go 1.24 中以实验特性登场的 testing/synctest 包,在 Go 1.25 中正式毕业,成为标准库的一员。它为并发代码测试提供了前所未有的利器,通过虚拟化时间和调度,让编写可靠、无 flakiness 的并发测试成为可能。我曾撰写过一个征服 Go 并发测试的微专栏,系统地介绍了该包的设计与实践,欢迎大家订阅学习。

encoding/json/v2 开启实验

这是 Go 1.25 最受关注的实验性特性之一!通过 GOEXPERIMENT=jsonv2 环境变量,我们可以启用一个全新的、高性能的 JSON 实现。

Go 1.25 includes a new, experimental JSON implementation… The new implementation performs substantially better than the existing one under many scenarios.

根据官方说明,json/v2 在解码性能上相较于 v1 有了“巨大”的提升。这是 Go 社区多年来对 encoding/json 包性能诟病的一次正面回应。虽然其 API 仍在演进中,但它预示着 Go 的 JSON 处理能力未来将达到新的高度。对 v2 的初探,可以参考我的文章《手把手带你玩转 GOEXPERIMENT=jsonv2:Go 下一代 JSON 库初探》。jsonv2支持真流式编解码的方法,也可以参考《Go json/v2实战:告别内存爆炸,掌握真流式Marshal和Unmarshal》这篇文章。

sync.WaitGroup.Go:并发模式更便捷

Go 语言的并发编程哲学之一就是让事情保持简单。Go 1.25 在 sync.WaitGroup 上新增的 Go 方法,正是这一哲学的体现。

这个新方法旨在消除 wg.Add(1) 和 defer wg.Done() 这一对经典的样板代码。现在,你可以直接调用 wg.Go(func() { … }) 来启动一个被 WaitGroup 追踪的 goroutine,Add 和 Done 的调用由 Go 方法在内部自动处理。这不仅让代码更简洁,也从根本上避免了因忘记调用 Add 或 Done 而导致的常见并发错误。

关于这个便捷方法的来龙去脉和设计思考,可以回顾我之前的文章《WaitGroup.Go 要来了?Go 官方提案或让你告别 Add 和 Done 样板代码》。

其他:Trace Flight Recorder

最后,我想特别提一下 runtime/trace 包新增的 Flight Recorder API。传统的运行时 trace 功能强大但开销巨大,不适合在生产环境中持续开启。

trace.FlightRecorder 提供了一种轻量级的解决方案:它将 trace 数据持续记录到一个内存中的环形缓冲区。当程序中发生某个重要事件(如一次罕见的错误)时,我们可以调用 FlightRecorder.WriteTo 将最近一段时间的 trace 数据快照保存到文件。这种“事后捕获”的模式,使得在生产环境中调试偶发、疑难的性能或调度问题成为可能,是 Go 诊断能力的一次重大升级。更多详情可以参阅《Go pprof 迎来重大革新:v2 提案详解,告别默认注册,拥抱飞行记录器》。

小结

Go 1.25 的发布,再次彰显了 Go 语言务实求进的核心哲学。它没有追求华而不实的语法糖,而是将精力聚焦于那些能为广大开发者带来“无形收益”的领域:更智能的运行时、更快的 GC、更可靠的编译器、更高效的工具链

这些看似底层的改进,正是 Go 作为一门“生产力语言”的价值所在。它让开发者可以专注于业务逻辑,而将复杂的系统优化和环境适配,放心地交给 Go 语言自身。

我鼓励大家尽快将 Go 1.25 应用到自己的项目中,亲自感受这些变化带来的提升。Go 的旅程,仍在继续,让我们共同期待它在未来创造更多的可能。

感谢阅读!

如果这篇文章让你对 Go 1.25 新特性有了新的认识,请帮忙 点赞分享,让更多朋友一起学习和进步!


你的Go技能,是否也卡在了“熟练”到“精通”的瓶颈期?

  • 想写出更地道、更健壮的Go代码,却总在细节上踩坑?
  • 渴望提升软件设计能力,驾驭复杂Go项目却缺乏章法?
  • 想打造生产级的Go服务,却在工程化实践中屡屡受挫?

继《Go语言第一课》后,我的《Go语言进阶课》终于在极客时间与大家见面了!

我的全新极客时间专栏 《Tony Bai·Go语言进阶课》就是为这样的你量身打造!30+讲硬核内容,带你夯实语法认知,提升设计思维,锻造工程实践能力,更有实战项目串讲。

目标只有一个:助你完成从“Go熟练工”到“Go专家”的蜕变! 现在就加入,让你的Go技能再上一个新台阶!


商务合作方式:撰稿、出书、培训、在线课程、合伙创业、咨询、广告合作。如有需求,请扫描下方公众号二维码,与我私信联系。

如发现本站页面被黑,比如:挂载广告、挖矿等恶意代码,请朋友们及时联系我。十分感谢! Go语言第一课 Go语言进阶课 AI原生开发工作流实战 Go语言精进之路1 Go语言精进之路2 Go语言第一课 Go语言编程指南
商务合作请联系bigwhite.cn AT aliyun.com

欢迎使用邮件订阅我的博客

输入邮箱订阅本站,只要有新文章发布,就会第一时间发送邮件通知你哦!

这里是 Tony Bai的个人Blog,欢迎访问、订阅和留言! 订阅Feed请点击上面图片

如果您觉得这里的文章对您有帮助,请扫描上方二维码进行捐赠 ,加油后的Tony Bai将会为您呈现更多精彩的文章,谢谢!

如果您希望通过微信捐赠,请用微信客户端扫描下方赞赏码:

如果您希望通过比特币或以太币捐赠,可以扫描下方二维码:

比特币:

以太币:

如果您喜欢通过微信浏览本站内容,可以扫描下方二维码,订阅本站官方微信订阅号“iamtonybai”;点击二维码,可直达本人官方微博主页^_^:
本站Powered by Digital Ocean VPS。
选择Digital Ocean VPS主机,即可获得10美元现金充值,可 免费使用两个月哟! 著名主机提供商Linode 10$优惠码:linode10,在 这里注册即可免费获 得。阿里云推荐码: 1WFZ0V立享9折!


View Tony Bai's profile on LinkedIn
DigitalOcean Referral Badge

文章

评论

  • 正在加载...

分类

标签

归档



View My Stats