标签 泛型 下的文章

Go 语言之父亲自下场道歉:藏在 Spec 里的十年“笔误”,终于要修正了!

本文永久链接 – https://tonybai.com/2026/03/25/go-spec-contradiction-in-types-section

大家好,我是Tony Bai。

在 Go 语言的世界里,type 是我们每天都在打交道的关键字。但如果我今天问你一个极其基础的问题:

Go 语言内置的 bool 类型,到底是不是一个“Defined Type(已定义类型)”?

你可能会愣一下,然后不假思索地回答:“那必须是啊,bool 是语言自带的,当然是已定义的。”

但如果我再追问一句:“既然它是 Defined Type,为什么我们不能给它绑定方法,像 func (b bool) IsTrue() {} 这样写?”

你可能就彻底懵了。

别慌,如果你对这个问题感到困惑,说明你已经触及到了 Go 语言类型系统设计中最深、也最容易被忽视的一个“历史遗留问题”。

就在最近,Go 官方 GitHub 仓库中,一个看似在“抠字眼”的 Issue #78208(spec: contradiction in Types section) 引来了社区里多位Go开发者下场激烈辩论,最终甚至连 Go 语言三巨头之一、被誉为“Go 语言之父之一”的 Robert Griesemer 都亲自现身,发表了一段长文来“认错”,并用拉丁语写下了那句沉重的 “Mea culpa”(我的锅)

今天,我们就来当一次“技术侦探”,顺着这个 Issue 的蛛丝马迹,硬核扒开 Go 语言规范(Spec)的底层,看看这个小小的 bool 类型背后,到底藏着 Go 团队一段怎样的设计“原罪”,以及它对我们日常编码产生了多大的深远影响。

一段自相矛盾的官方“圣经”

故事的起因非常简单。一位开发者在精读 Go 语言官方规范(Spec,被誉为 Go 语言的“圣经”)时,发现了一个极其明显的逻辑矛盾。

Types 章节,规范明确地将“具名类型(Named types)”分为了三类:

“Predeclared types, defined types, and type parameters are called named types.”
(预声明类型、已定义类型和类型参数,统称为具名类型。)

这里的措辞,清晰地将这三者并列。

但当你翻到 Boolean types 章节时,却赫然写着:

“The predeclared boolean type is bool; it is a defined type.”
(预声明的布尔类型是 bool;它是一个已定义类型。)

矛盾爆发了!

如果“预声明类型”和“已定义类型”是平级的、不同的两个分类,那 bool 怎么可能既是前者,又是后者?这就像生物分类学里说“哺乳动物和爬行动物是不同的两个纲”,然后又说“老虎是一种爬行动物”一样荒谬。

这个问题瞬间在社区里炸开了锅。

一场关于“定义”的思辨

Issue 下方的评论区,堪称一场神仙打架。

一部分开发者认为这是明显的 Spec 笔误。他们旗帜鲜明地指出:

“bool 不是一个已定义类型。因为它不能拥有方法。对于一个已定义类型 T,它必须出现在 type T … 的定义中。”

这话说得掷地有声。我们都知道,type MyInt int 之后,MyInt 才是一个真正的 Defined Type,我们可以给它绑定方法。而 bool 显然不符合这个特征。

但另一派开发者,也开始了精彩的“诡辩”。他们认为:

“Spec 并没有说这三个分类是互斥的。‘预声明’只是意味着这个类型是编译器内置的,但它本质上依然是一个‘已定义’的类型。只不过它的定义对我们不可见罢了。”

双方你来我往,从类型的方法集,辩论到 Go 1.9 引入类型别名(Type Alias)时的历史背景,再到 Go 1.18 引入泛型后对“具名类型”的重新定义。

就在大家争得面红耳赤之时,Go 语言之父之一 Robert Griesemer 悄然现身,一锤定音。

Go 语言类型系统的“原罪”

Robert Griesemer 的长篇回复,像一本尘封已久的历史档案,为我们揭开了 Go 语言在类型设计上的一段“黑历史”。

他首先承认:“没错,你们都被搞糊涂了。这个 Spec 写得确实有歧义,我们马上就改。”

然后,他开始讲述这个“小小的”用词不当背后,隐藏的 Go 团队在设计类型系统时的“原罪”。

原罪的根源:Go 团队混淆了“拥有名字”和“拥有唯一身份”这两个概念。

  1. Go 1.0 时代

那时只有“具名类型”和“匿名类型”。为了让 int、bool 这些内置类型拥有独一无二的身份(Type Identity),Go 团队很自然地把它们也归入了“具名类型”,毕竟它们确实有名字。这在当时看起来很完美。

  1. Go 1.9 时代(引入类型别名)

type NewString = string 这样的类型别名出现了。NewString 也有名字,但它的身份和 string 是完全一样的。这就和原来的“具名=唯一身份”的假设冲突了。

为了解决这个问题,Go 团队做了一个现在看来极其糟糕的决定:他们把原来表示“唯一身份”的“具名类型”,改名为了 “已定义类型(Defined Type)”。而 bool、int 这些内置类型,为了保留它们的唯一身份,也就跟着一起被“定义”成了 Defined Type。

  1. Go 1.18 时代(引入泛型)

类型参数 T 出现了。T 也有名字,而且不同的类型参数(比如 T 和 P)必须拥有不同的身份。于是,Go 团队不得不又把“具名类型(Named Type)”这个概念重新捡了回来,这次用它来统称所有“拥有唯一身份”的类型。

看懂了吗?

bool 之所以被错误地描述为 defined type,完全是一次历史的意外。它是 Go 团队在不断给语言打补丁、修补旧概念的过程中,留下的一块“历史伤疤”。

Robert Griesemer 最后感慨道:“Mea culpa(我的锅)。”

这个小小的用词问题,背后是 Go 语言设计者在面对一个不断演进的复杂系统时,所做出的艰难权衡与无奈妥协。

他甚至自嘲般地补了一刀:

“为了让你们更受伤一点,我再告诉你们一个秘密:预声明的 any 类型,其实根本不是一个具名类型,它只是匿名接口 interface{} 的一个别名。”

最后,我们看到了Robert Griesemer 提交了一个cl,给出了修改方案:在spec中明确”predeclared types are named, not defined types”,即预声明类型是具名类型,但不是已定义类型。同时加上了对 any 这个预声明类型不是具名类型的澄清。

这个“抠字眼”的争论,对我们写代码有何意义?

看到这里,你可能会觉得:“搞了半天,不就是改几个英文单词吗?关我写业务代码什么事?”

关系太大了。理解了这段“黑历史”,你才能真正打通 Go 类型系统的任督二脉,尤其是在处理泛型和接口时。

1. 你才能真正理解“类型约束”的本质。

在泛型函数中,~string 这个约束,匹配的是所有底层类型为 string 的类型。它包含了 string 本身,也包含了 type MyString string 这种 Defined Type。

但如果你只写 string,那么 MyString 类型的变量是传不进去的。

因为 string 是“预声明类型”,而 MyString 是“已定义类型”,尽管底层结构一样,但它们的“身份”在 Go 的世界里是完全不同的。

2. 你才能彻底搞懂“方法集”的规则。

为什么 bool 不能有方法?因为它不是通过 type 关键字在你的代码里定义的。方法只能绑定在你明确定义的类型上。这个规则,是 Go 语言不允许你“污染”内置类型的安全护栏。

3. 你才能在写库时,做出更高级的 API 设计。

当你设计一个库的 API 时,到底是应该接受 string,还是应该接受 interface{ String() string }?

如果你只接受 string,那么所有基于 string 定义的新类型都必须强制转换,非常不便。

但如果你接受接口,就意味着你放弃了对底层数据结构的强约束。

理解了“预声明类型”与“已定义类型”在身份上的本质区别,你才能在这两者之间做出最合理的架构权衡。

小结:于细微处,见真章

一个看似吹毛求疵的 Issue,最终牵扯出了 Go 语言长达十几年的演进历史和设计哲学。

它告诉我们: 一门伟大的编程语言,并不是一蹴而就的天才设计,而是在无数次的妥协、修补和自我反思中,不断螺旋上升的有机生命体。

而我们作为开发者,对这门语言最好的尊重,就是不仅要知其然,更要知其所以然。

下次当你在面试中被问到 Go 的类型系统时,不妨把这个关于 bool 的故事讲给面试官听。相信我,这远比你背诵一百遍枯燥的语法规则,更能证明你对这门语言的深刻理解。

资料链接:

  • https://github.com/golang/go/issues/78208
  • https://go-review.googlesource.com/c/go/+/757120

今日互动探讨

在你的 Go 开发生涯中,遇到过哪些让你对 Go 的类型系统感到极其困惑,甚至怀疑人生的场景?比如类型断言的 panic、空接口的转换、还是泛型的约束?

欢迎在评论区分享你的踩坑经历!


还在为“复制粘贴喂AI”而烦恼?我的新专栏 AI原生开发工作流实战 将带你:

  • 告别低效,重塑开发范式
  • 驾驭AI Agent(Claude Code),实现工作流自动化
  • 从“AI使用者”进化为规范驱动开发的“工作流指挥家”

扫描下方二维码,开启你的AI原生开发之旅。


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

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

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

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

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


原「Gopher部落」已重装升级为「Go & AI 精进营」知识星球,快来加入星球,开启你的技术跃迁之旅吧!

我们致力于打造一个高品质的 Go 语言深度学习AI 应用探索 平台。在这里,你将获得:

  • 体系化 Go 核心进阶内容: 深入「Go原理课」、「Go进阶课」、「Go避坑课」等独家深度专栏,夯实你的 Go 内功。
  • 前沿 Go+AI 实战赋能: 紧跟时代步伐,学习「Go+AI应用实战」、「Agent开发实战课」、「Agentic软件工程课」、「Claude Code开发工作流实战课」、「OpenClaw实战分享」等,掌握 AI 时代新技能。
  • 星主 Tony Bai 亲自答疑: 遇到难题?星主第一时间为你深度解析,扫清学习障碍。
  • 高活跃 Gopher 交流圈: 与众多优秀 Gopher 分享心得、讨论技术,碰撞思想火花。
  • 独家资源与内容首发: 技术文章、课程更新、精选资源,第一时间触达。

衷心希望「Go & AI 精进营」能成为你学习、进步、交流的港湾。让我们在此相聚,享受技术精进的快乐!欢迎你的加入!

img{512x368}


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

别再像 2015 年那样写 Go 了:Modern Go 终极进化指南

本文永久链接 – https://tonybai.com/2026/03/02/modern-go-evolution-guide-1-0-to-1-26

大家好,我是Tony Bai。

Go 语言在业界最著名的标签之一就是“向后兼容承诺(Go 1 Compatibility Promise)”。一份 10 年前写下的 Go 1.4 代码,在今天的 Go 1.26 编译器下依然能完美编译并运行。

但这带来了一个副作用:许多 Go 开发者的思维和编码习惯,也停留在过去的时代。

我们依然能看到满天飞的 interface{}、冗长易错的 for 循环切片查找、为了获取指针而被迫抽离的辅助函数,以及在并发测试中繁琐的 Context 初始化。

近日,JetBrains 开源了一个名为 use-modern-go 的 AI Coding Agent Skill。这份Skill文件通过精准的 Prompt,强迫 AI 智能体在生成 Go 代码时,必须根据项目 go.mod 的版本,使用该版本支持的最现代化、最优雅的语法和标准库

这份文件简直是一座宝库!它不仅是给 AI 看的指令,更是给每一位 Gopher 的“代码现代化”体检表。

本文将以这份资料为基础,全面盘点从 Go 1.0 一路演进到 Go 1.26 的 Modern Go 特性。我们将通过清晰的 Before / After 对比示例,带你洗礼一遍 Go 语言的现代化之美。

第一阶段:早期的代码净化(Go 1.0 – Go 1.19)

虽然是早期版本,但这些 API 的引入确立了 Go 代码“少即是多”的审美基调。

时间的优雅流逝 (time.Since / time.Until)

在计算耗时或剩余时间时,不要再手动做减法了。

❌ Before (Legacy):

start := time.Now()
// do work
elapsed := time.Now().Sub(start)

deadline := time.Now().Add(5 * time.Second)
remaining := deadline.Sub(time.Now())

✅ After (Modern – Go 1.0/1.8):

start := time.Now()
// do work
elapsed := time.Since(start)

deadline := time.Now().Add(5 * time.Second)
remaining := time.Until(deadline)

错误处理的革命 (errors.Is)

Go 1.13 引入了错误包装(Error Wrapping)。使用 == 判断错误已经不再安全。

❌ Before (Legacy):

if err == sql.ErrNoRows {
    // 无法捕获 fmt.Errorf("query failed: %w", sql.ErrNoRows) 包装后的错误
}

✅ After (Modern – Go 1.13):

if errors.Is(err, sql.ErrNoRows) {
    // 即使被多层 %w 包装,依然能准确识别
}

告别 interface{} (any)

Go 1.18 引入了泛型,同时带来了一个赏心悦目的类型别名 any。

❌ Before (Legacy):

func PrintAll(vals[]interface{}) { ... }

✅ After (Modern – Go 1.18):

func PrintAll(vals[]any) { ... }

字符串无损切割 (strings.Cut / bytes.Cut)

解析键值对是最常见的操作。过去我们需要 strings.Index 配合切片操作,极易引发 panic: slice bounds out of range。

❌ Before (Legacy):

idx := strings.Index(header, ":")
if idx != -1 {
    key := header[:idx]
    value := header[idx+1:]
}

✅ After (Modern – Go 1.18):

if key, value, found := strings.Cut(header, ":"); found {
    // 安全、直观、一次调用
}

高性能字符串追加 (fmt.Appendf)

Go 1.19 引入了直接向字节切片追加格式化字符串的能力,避免了 fmt.Sprintf 带来的隐式内存分配。

❌ Before (Legacy):

buf := []byte("Prefix: ")
buf = append(buf,[]byte(fmt.Sprintf("user_id=%d", id))...) // 发生堆分配

✅ After (Modern – Go 1.19):

buf :=[]byte("Prefix: ")
buf = fmt.Appendf(buf, "user_id=%d", id) // 零分配(如果 buf 容量充足)

类型安全的原子操作 (atomic.Bool/Int64/Pointer)

放弃 atomic.Value 和难记的 atomic.StoreInt32 吧,Go 1.19 的泛型原子类型既安全又易读。

❌ Before (Legacy):

var flag int32 // 0: false, 1: true
atomic.StoreInt32(&flag, 1)
if atomic.LoadInt32(&flag) == 1 { ... }

✅ After (Modern – Go 1.19):

var flag atomic.Bool
flag.Store(true)
if flag.Load() { ... }

var cfg atomic.Pointer[Config]
cfg.Store(&Config{})

第二阶段:标准库的泛型文艺复兴(Go 1.20 – Go 1.21)

在这个阶段,经过两个大版本打磨的 Go 泛型,彻底释放了泛型的潜力,引入了大量期待已久的内置函数和集合操作库。

明确的克隆 (strings.Clone / bytes.Clone)

当你想持有一个大字符串/字节切片的极小一部分,又不想让垃圾回收器保留整个底层大数组时,你需要 Clone。

❌ Before (Legacy):

// 丑陋的黑魔法来强制复制字符串
copiedStr := string([]byte(hugeString[:10]))

✅ After (Modern – Go 1.20):

copiedStr := strings.Clone(hugeString[:10])

溯源 Context 取消原因 (context.WithCancelCause)

Context 被取消了,但究竟是因为超时、主动取消,还是底层的网络错误?Go 1.20 让你能够携带取消原因。

❌ Before (Legacy):

ctx, cancel := context.WithCancel(parent)
// 发生错误时
cancel()
// 其他协程只知道 ctx.Err() == context.Canceled

✅ After (Modern – Go 1.20):

ctx, cancel := context.WithCancelCause(parent)
// 发生错误时
cancel(fmt.Errorf("db connection lost"))

// 消费端可以查明真凶
err := context.Cause(ctx) // 返回 "db connection lost"

(注:Go 1.21 还补充了 context.WithTimeoutCause)

内置的魔法:min, max, clear

这是 Go 1.21 最受欢迎的内置函数。

❌ Before (Legacy):

// 求最大值(非浮点数只能自己写 if/else)
m := a
if b > m { m = b }

// 清空 Map
for k := range myMap { delete(myMap, k) }

✅ After (Modern – Go 1.21):

m := max(a, b) // 支持所有可比较类型
clear(myMap)   // 高效清空 map,保留底层容量

强大的 slices 和 maps 库

告别手动写 for 循环查找元素的日子。

❌ Before (Legacy):

func contains(list[]string, target string) bool {
    for _, v := range list {
        if v == target { return true }
    }
    return false
}

✅ After (Modern – Go 1.21):

import "slices"
import "maps"

// 查找
found := slices.Contains(items, target)
idx := slices.IndexFunc(users, func(u User) bool { return u.ID == 42 })

// 排序 (原 sort.Slice 需要写繁琐的 Less 函数)
slices.Sort(ints)
slices.SortFunc(users, func(a, b User) int { return cmp.Compare(a.Age, b.Age) })

// 紧凑与裁剪
items = slices.Compact(items) // 移除连续重复元素
items = slices.Clip(items)    // 移除切片多余的 capacity

// 字典操作
clonedMap := maps.Clone(originalMap)
maps.DeleteFunc(m, func(k string, v int) bool { return v < 0 })

更聪明的单次执行 (sync.OnceFunc / OnceValue)

sync.Once 很好用,但如果我们想只初始化一次并返回一个值,过去需要闭包外变量和额外的锁。

❌ Before (Legacy):

var once sync.Once
var config *Config
func GetConfig() *Config {
    once.Do(func() { config = loadConfig() })
    return config
}

✅ After (Modern – Go 1.21):

// 声明即完成包装,线程安全且优雅
var GetConfig = sync.OnceValue(func() *Config {
    return loadConfig()
})

// 使用
cfg := GetConfig()

第三阶段:语法细节与 Web 路由的飞跃(Go 1.22)

整数范围循环 (for i := range n)

❌ Before: for i := 0; i < 10; i++ { … }
✅ After: for i := range 10 { … }

默认值救星 (cmp.Or)

返回第一个非零值,简直是读取环境变量的神器。

❌ Before (Legacy):

port := os.Getenv("PORT")
if port == "" {
    port = "8080"
}

✅ After (Modern – Go 1.22):

port := cmp.Or(os.Getenv("PORT"), "8080")

史诗级加强的 http.ServeMux

标准库路由器终于支持 HTTP 方法和路径参数了,很多小项目再也不需要引入 gin 或 chi。

✅ Modern Go 1.22:

mux := http.NewServeMux()
mux.HandleFunc("POST /api/users/{id}", func(w http.ResponseWriter, r *http.Request) {
    userID := r.PathValue("id")
    // ...
})

第四阶段:迭代器时代的黎明(Go 1.23 – Go 1.24)

Go 1.23 引入了 iter.Seq(迭代器),这是自泛型以来最大的范式转变。它统一了所有“序列”的遍历方式。

迭代器与切片/字典的梦幻联动

提取 map 的所有 key 并排序,过去需要手动 append 加 sort。

✅ Modern Go 1.23:

// 获取字典的 keys 迭代器 -> 收集为切片 -> 返回新切片
keys := slices.Collect(maps.Keys(m))

// 收集并一步排序
sortedKeys := slices.Sorted(maps.Keys(m))

测试和基准测试的现代化 (t.Context(), b.Loop())

Go 1.24 对 testing 库进行了大规模重构,代码更加精简防错。

❌ Before (Legacy Testing):

func TestFoo(t *testing.T) {
    ctx, cancel := context.WithCancel(context.Background())
    defer cancel()
    doSomething(ctx)
}

func BenchmarkBar(b *testing.B) {
    for i := 0; i < b.N; i++ {
        doWork() // 编译器可能会过度优化这里的代码
    }
}

✅ After (Modern Go 1.24):

func TestFoo(t *testing.T) {
    // 自动随测试结束而取消的 Context
    ctx := t.Context()
    doSomething(ctx)
}

func BenchmarkBar(b *testing.B) {
    // 防止编译器优化掉内部逻辑的全新循环方式
    for b.Loop() {
        doWork()
    }
}

JSON 标签终极补丁 (omitzero)

长久以来,JSON 的 omitempty 标签对 time.Time 和嵌套 struct 这种“非空即零”的类型无效(因为它们永远不是 nil)。Go 1.24 终于引入了 omitzero。

✅ Modern Go 1.24:

type User struct {
    // 以前:即使时间是 0001-01-01 也会被序列化输出
    // 现在:只要是零值,就忽略
    CreatedAt time.Time json:"created_at,omitzero"
}

零分配迭代分割 (strings.SplitSeq)

当你只需要遍历分割后的字符串,而不需要将其存入切片时,迭代器能帮你省下所有内存分配。

  • ❌ Before (Allocates memory): for _, part := range strings.Split(s, “,”) { … }
  • ✅ After (Zero allocation): for part := range strings.SplitSeq(s, “,”) { … }

第五阶段:属于现在的未来(Go 1.25 – Go 1.26)

让我们来看看最近两个发布版本实装的黑科技。

拯救 WaitGroup (wg.Go())

Go 1.25 消除了并发控制中最常见的 Bug:忘记写 wg.Add(1) 或者忘记 defer wg.Done()。

❌ Before (Legacy):

var wg sync.WaitGroup
for _, item := range items {
    wg.Add(1)
    go func(i Item) {
        defer wg.Done()
        process(i)
    }(item)
}
wg.Wait()

✅ After (Modern Go 1.25):

var wg sync.WaitGroup
for _, item := range items {
    // 自动处理 Add(1) 和内部的 Done(),连闭包变量捕获问题都不用再担心
    wg.Go(func() {
        process(item)
    })
}
wg.Wait()

指针获取的终极解法 (new(expr))

在 Go 1.26 中,为了给结构体的指针字段(常见于 Protobuf/JSON 生成的代码)赋值,你再也不需要写恶心的辅助函数了。new() 终于支持了表达式。

❌ Before (Legacy):

timeout := 30
debug := true
cfg := Config{
    Timeout: &timeout, // 必须先单独声明变量
    Debug:   &debug,
}

✅ After (Modern Go 1.26):

cfg := Config{
    Timeout: new(30),   // 推断为 *int
    Debug:   new(true), // 推断为 *bool
    Role:    new("admin"), // *string
}

警告:请直接写 new(30),千万不要写 new(int(30)) 这种脱裤子放屁的类型转换,编译器足够聪明。

泛型安全类型断言 (errors.AsType)

处理自定义错误时,errors.As 极易用错,因为它需要传入一个指针的指针,如果传入非指针会在运行时直接 Panic。Go 1.26 用泛型完美解决了它。

❌ Before (Unsafe):

var pathErr *os.PathError
// 极易漏写 & 导致 panic
if errors.As(err, &pathErr) {
    handle(pathErr)
}

✅ After (Modern Go 1.26):

// 编译期类型安全,返回具体的实例
if pathErr, ok := errors.AsType[*os.PathError](err); ok {
    handle(pathErr)
}

小结:让 AI 成为代码现代化的推手

回顾这从 Go 1.0 到 1.26 的演进史,我们看到了一条清晰的脉络:Go 官方正在极力消除样板代码(Boilerplate),同时坚定地维持着语言的简单与直白。

JetBrains 开源的这个 use-modern-go Skill 给了我们一个绝佳的启示:在 AI 编程时代,不要让大模型去学习网上那些陈旧的、十年前的 StackOverflow 答案。 通过系统性的 Prompt 引导,我们可以强迫 AI 写出最符合当前语言版本的、最高效的 Modern Code。

作为 Gopher,是时候给你的脑海中的“Go 语言编译器”升个级了。下一次敲下代码时,问问自己:“这是 2015 年的写法,还是 2026 年的写法?”


你的代码里还有“老古董”吗?

哪怕 Go 1.26 已经发布,很多人的 go.mod 依然停留在 1.16 甚至更早。在这些 Modern 特性中,哪一个最让你感到“相见恨晚”?你在重构老代码时,遇到过哪些由于“兼容性思维”导致的阻碍?

欢迎在评论区分享你的 Modern Go 实践!


还在为“复制粘贴喂AI”而烦恼?我的新专栏 AI原生开发工作流实战 将带你:

  • 告别低效,重塑开发范式
  • 驾驭AI Agent(Claude Code),实现工作流自动化
  • 从“AI使用者”进化为规范驱动开发的“工作流指挥家”

扫描下方二维码,开启你的AI原生开发之旅。


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

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

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

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

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


想系统学习Go,构建扎实的知识体系?

我的新书《Go语言第一课》是你的首选。源自2.4万人好评的极客时间专栏,内容全面升级,同步至Go 1.24。首发期有专属五折优惠,不到40元即可入手,扫码即可拥有这本300页的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