标签 GC 下的文章

从arena、memory region到runtime.free:Go内存管理探索的务实转向

本文永久链接 – https://tonybai.com/2025/09/18/go-runtime-free-proposal

大家好,我是Tony Bai。

Go 的垃圾收集器(GC)是其简单性和并发安全性的基石,但也一直是性能优化的焦点。近年来,Go 核心团队为了进一步降低 GC 开销,进行了一系列前沿探索:从备受争议的arena 实验,到更优雅但实现复杂的 memory regions构想,最终,焦点似乎汇聚在了一项更务实、更具潜力的提案上——runtime.free。这项编号为 #74299 的实验性提案,正试图为 Go 的内存管理引入一个革命性的新维度:允许编译器和部分标准库在特定安全场景下,绕过 GC,直接释放和重用内存。其原型已在 strings.Builder 等场景中展现出高达 2 倍的性能提升。

本文将带着大家一起回顾 Go 内存管理的这段探索之旅,并初步剖析一下 runtime.free 提案的背景、核心机制及其对 Go 性能生态的深远影响。

背景:一场关于“手动”内存管理的漫长探索

Go 语言自诞生以来,其自动内存管理(GC)一直是核心特性之一。然而,对于性能极致敏感的场景——例如高吞吐量的网络服务——GC 的开销始终是开发者关注的焦点。为了赋予开发者更多控制力,Go 团队近年来开启了一系列关于“手动”或“半自动”内存管理的探索。

第一站:arena 实验——功能强大但难以融合

arena 实验(#51317)是第一次大胆的尝试。它引入了一个 arena.Arena 类型,允许开发者将一组生命周期相同的对象分配到一个独立的内存区域中,并在不再需要时一次性、批量地释放整个区域

  • 优点:arena 在特定场景下取得了显著的性能提升,因为它极大地减少了 GC 的扫描和回收工作。
  • 问题:arena 的 API 侵入性太强。几乎所有需要利用 arena 的函数都必须额外接收一个 arena 参数,这会导致 API 的“病毒式”传播,并且与 Go 的隐式接口、逃逸分析等特性组合得非常糟糕。最终,由于其糟糕的“可组合性”,arena 提案被无限期搁置。

第二站:memory regions——更优雅的构想与巨大的挑战

吸取了 arena 的教训,Go 团队提出了一个更优雅、更符合 Go 哲学的构想:内存区域(Memory Regions)(#70257)。其核心思想是,通过一个 region.Do(func() { … }) 调用,将一个函数作用域内的所有内存分配隐式地绑定到一个临时的、与 goroutine 绑定的区域中。

  • 优点:API 对用户透明,无需修改现有函数的签名。更重要的是,它是内存安全的——如果区域内的某个对象“逃逸”到了区域之外,运行时会自动将其“拯救”出来,交还给全局 GC 管理,避免了 arena 可能导致的 use-after-free 崩溃。
  • 问题:这个优雅设计的背后,是极其复杂的实现。它需要在开启区域的 goroutine 中启用一个特殊的、低开销的写屏障(write barrier)来动态追踪内存的逃逸。虽然理论上可行,但其实现复杂度和潜在的性能开销,使其成为一个长期且充满不确定性的研究课题。

最终的焦点:runtime.free——务实且精准的“外科手术”

在 arena 的侵入性和 memory regions 的复杂性之间,Go 团队似乎找到了一个更务实、更具工程可行性的平衡点——runtime.free 提案。

它不再追求一个“要么全有,要么全无”的全局解决方案,而是提出了一种精准的、由编译器和运行时主导的“外科手术”。其核心思想是:与其让开发者手动管理整个内存区域,不如让更了解代码细节的编译器底层标准库,在绝对安全的前提下,对那些生命周期短暂的、已知的堆分配进行点对点的、即时的释放和重用

这种方法解决了 arena 的可组合性问题(因为它是自动的或内部的),也绕开了 memory regions 的全局复杂性。它像一把锋利的手术刀,精确地切除了那些最明确、最高频的冗余内存分配,为解决 Go 性能优化中的“鸡与蛋”问题提供了全新的思路。

runtime.free 的双重策略:编译器自动化与标准库手动优化

该提案并非要将 free 的能力直接暴露给普通开发者。相反,它采取了一种高度受控的、分两路进行的策略:

1. 编译器自动化 (runtime.freetracked)

这是该提案最激动人心的部分。编译器将获得自动插入内存跟踪和释放代码的能力。

  • 工作流程

    1. 识别:当编译器遇到一个 make([]T, size),它能证明这个 slice 的生命周期不会超过当前函数作用域,但因其大小未知(或超过 32 字节)而必须在堆上分配时,它会将这次分配标记为“可跟踪”。
    2. 跟踪:编译器会生成 makeslicetracked64 来分配内存,并将一个“跟踪对象”记录在当前函数栈上的一个特殊数组 freeablesArr 中。
    3. 释放:编译器会自动插入一个 defer freeTracked(&freeables) 调用。当函数退出时,这个 defer 会被执行,通知运行时可以安全地回收 freeablesArr 中记录的所有堆对象。
  • 对开发者的影响:这意味着,未来开发者编写的许多看似会产生堆分配的函数,将被编译器自动重写为不产生 GC 压力的版本,而开发者对此完全无感

// 开发者编写的代码
func f1(size int) {
    s := make([]int64, size) // 堆分配
    // ... use s
}

// 编译器可能重写为(概念上)
func f1(size int) {
    var freeablesArr [1]trackedObj
    freeables := freeablesArr[:]
    defer runtime.freeTracked(&freeables)

    s := runtime.makeslicetracked64(..., &freeables) // 分配并跟踪
    // ... use s
}

2. 标准库手动优化 (runtime.freesized)

对于一些底层、性能关键的标准库组件,它们内部的内存管理逻辑比编译器能静态证明的要复杂。对于这些场景,提案提供了一个受限的、手动的 runtime.freesized 接口。

  • 目标场景

    • strings.Builder / bytes.Buffer 的扩容:当内部 []byte 缓冲区需要扩容时,旧的、较小的缓冲区就可以被立即释放。
    • map 的扩容:当 map 增长或分裂时,旧的 backing array 也可以被回收。
    • slices.Collect:在构建最终 slice 过程中产生的中间 slice 也可以被释放。
  • 惊人的性能提升:提案中的基准测试显示,通过在 strings.Builder 的扩容逻辑中手动调用 runtime.freesized,在有多次写入(即多次扩容)的场景下,其性能提升了 45% 到 55%,几乎是原来的两倍快!

这证明,在正确的“热点”位置进行手动释放,可以带来巨大的性能收益。

性能影响与权衡

引入手动内存管理,必然会带来对正常分配路径的性能影响。提案对此进行了细致的评估:

  • 对正常分配路径的影响:基准测试表明,即使开启了 runtimefree 实验,对于不涉及内存重用的普通分配路径,其性能影响在 -1.5% 到 +2.2% 之间,几何平均值几乎为零。这表明该功能在不使用时,几乎是“免费”的。
  • 潜在的性能收益
    1. 减少 GC CPU 使用:这是最直接的好处。
    2. 延长 GC 周期:更少的垃圾意味着 GC 运行频率更低,从而减少写屏障(write barrier)开启的时间,提升应用代码的执行速度。
    3. 更优的缓存局部性:被释放的内存可以立即被下一个分配重用,可能形成 LIFO(后进先出)式的内存访问模式,对 CPU 缓存极为友好。
    4. 减少 GC 停顿:更少的 GC 工作意味着更少的 STW(Stop-The-World)时间和 GC 辅助(assist)开销。

小结:Go 内存管理的“第三条路”

runtime.free 提案并非要将 Go 变成 C++ 或 Rust,它无意将手动内存管理的复杂性抛给普通开发者。相反,它代表了 Go 在自动内存管理(GC)和静态内存管理(栈分配)之外,探索的“第三条路”——由编译器和运行时主导的、高度受控的动态内存优化

这一探索是务实且极具潜力的:

  • 务实:它从解决现实的性能瓶颈(如 strings.Builder)和优化僵局(逃逸分析)入手,目标明确。
  • 安全:通过将能力严格限制在编译器和少数底层标准库中,它最大限度地避免了困扰其他语言的手动内存管理错误。
  • 潜力巨大:一旦这个机制成熟,编译器可以将其应用到更多模式中(如循环内的 append),进一步减少 Go 程序的内存分配。

虽然这项工作仍处于实验阶段,但它清晰地指明了 Go 性能优化的下一个前沿方向。通过让编译器和运行时变得更加“智能”,在保证安全性的前提下,选择性地介入内存管理,Go 语言有望在保持其简洁易用性的同时,攀上新的性能高峰。

参考资料

  • runtime, cmd/compile: add runtime.free, runtime.freetracked and GOEXPERIMENT=runtimefree – https://github.com/golang/go/issues/74299
  • a safe free of memory proposal, runtime.FreeMemory() – https://groups.google.com/g/golang-nuts/c/cmpiArv10f4
  • Directly freeing user memory to reduce GC work – https://go.googlesource.com/proposal/+/94843c2c941f64a86001e51ed775b918cc89b365/design/74299-runtime-free.md
  • memory regions – https://github.com/golang/go/discussions/70257
  • proposal: arena: new package providing memory arenas – https://github.com/golang/go/issues/51317

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

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

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

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

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


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

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语言第一课 Go语言进阶课 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