标签 接口 下的文章

为什么 Go 社区强调避免不必要的抽象?—— 借用海德格尔哲学寻找“正确”的答案

本文永久链接 – https://tonybai.com/2026/01/16/go-community-the-right-kind-of-abstraction

大家好,我是Tony Bai。

“Go 的哲学强调避免不必要的抽象。”

这句话我们听过无数次。当你试图引入 ORM、泛型 Map/Reduce 、接口或者复杂的设计模式时,往往会收到这样的反馈。这句话本身没有错,但难点在于:到底什么是“不必要”的?

函数是抽象吗?汇编是抽象吗?如果不加定义地“避免抽象”,我们最终只能对着硅片大喊大叫。

在 GopherCon UK 2025 上,John Cinnamond 做了一场与众不同的演讲。他没有展示任何炫酷的并发模式,而是搬出了马丁·海德格尔(Martin Heidegger)和伊曼努尔·康德(Immanuel Kant),试图用哲学的视角,为我们解开关于 Go 抽象的终极困惑。

注:海德格尔与《存在与时间》

马丁·海德格尔(Martin Heidegger)是 20 世纪最重要的哲学家之一。他在 1927 年的巨著《存在与时间》(Being and Time) 中,深入探讨了人(此在)如何与世界互动。John Cinnamond 在演讲中引用的核心概念——“上手状态” (Ready-to-hand)“在手状态” (Present-at-hand),正是海德格尔用来描述我们与工具(如锤子)之间关系的术语。这套理论极好地解释了为什么优秀的工具(或代码抽象)应该是“透明”的,而糟糕的工具则会强行占据我们的注意力。

img{512x368}

我们都在使用的“必要”抽象

首先,让我们承认一个事实:编程本身就是建立在无数层抽象之上的。

  • 泛型:这是对类型的抽象。虽然 Go 曾长期拒绝它,但在技术上它是必要的,否则我们将充斥着重复代码。
  • 接口:这是对行为的抽象。io.Reader 让我们不必关心数据来自文件还是网络。
  • 函数:这是对指令序列的抽象。没有它,我们只能写长长的 main 函数。
  • 汇编语言:这是对机器码的抽象。

所以,当我们说“避免不必要的抽象”时,我们真正想表达的其实是——避免“不恰当” (Inappropriate) 的抽象

那么,如何判断一个抽象是否“恰当”?

何为抽象?—— 一场有目的的“细节隐藏”

在深入探讨“正确”的抽象之前,我们必须先回到最基本的定义。John Cinnamond 在演讲中给出了一个精炼而深刻的定义:

“抽象是一种表示 (Representation),但它是一种刻意移除被表示事物某些细节的表示。”

让我们拆解这个定义:

  1. 抽象是一种“表示”,而非事物本身
    它不是代码的实体,而是代码的地图或模型。例如,一辆模型汽车是真实汽车的表示,但 Gopher 吉祥物是地鼠的抽象——它刻意省略了真实地鼠的所有细节,只保留了核心特征。

  1. 抽象是“有目的的”细节移除
    这与仅仅是“不精确”或“粗糙”不同。抽象是有意为之的,它不试图精确描绘所有方面,而是只关注某个特定维度

  1. 抽象在编程中具有动态性
    • 不确定引用 (Indefinite Reference):一个抽象(如 io.Reader)通常可以指代许多不同的具体实现。
    • 开放引用 (Open Reference):抽象的内容或它所指代的事物可以随着时间而改变。

为什么要刻意移除细节?John 总结了几个核心动机:

  • 避免重复代码:将重复的逻辑提取到抽象中。
  • 统一不同的实现:允许以统一的方式处理本质上不同的数据结构(如所有实现了 Read 方法的类型)。
  • 推迟细节:隐藏那些当下不重要、或开发者不关心的细节(例如,你坐火车参会,不需要知道每节车厢的编号)。
  • 揭示领域概念:用抽象来更好地表达业务领域中的核心概念。
  • 驾驭复杂性:这是最核心的理由——没有抽象,我们无法在大脑中一次性处理所有细节,也就无法解决复杂的问题。

但请记住,并非所有抽象都是一样的。John 将它们分为三类:

  1. 基于“它是如何工作的” (How it works)
    这是为了代码复用而提取的抽象。例如,你发现两处代码都在做“检查用户是否是管理员”的逻辑,于是将其提取为一个函数。这种抽象关注的是内部机制。 (这类抽象通常比较脆弱,一旦实现细节变化,抽象可能就会失效。)

  2. 基于“它做了什么” (What it does)
    这是 Go 语言中接口(Interface)最典型的用法。例如 io.Reader,我们不关心它是文件还是网络连接,我们只关心它能“读取字节”。这是一种行为抽象。

  3. 基于“它是什么” (What it is)
    这是基于领域模型的抽象。例如一个 User 结构体,它代表了系统中的一个实体。这种抽象关注的是本质属性。

在现实中,好的抽象往往是这三者的混合体,但在设计时,明确你是在抽象“行为”还是“实现”,对于判断抽象的质量至关重要。

理解了抽象的本质,我们可能会觉得:既然抽象能驾驭复杂性,那是不是越多越好?

且慢。在急于评判一个抽象是否“恰当”之前,我们必须先意识到一个常被技术人员忽略的现实:抽象不仅存在于代码中,更存在于人与人的互动里。 这将我们引向了一个更现实的考量维度。

抽象的代价 —— 代码是写给人看的

John 提醒我们,软件开发本质上是一项社会活动 (Social Activity)

“除非你是为了自己写着玩,否则你的代码总是写给别人看的。团队是一个微型社会,它有自己的习俗、信仰和‘传说’(Lore)。”

引入一个新的抽象,本质上是在向这个微型社会引入一种新的文化或规则。这意味着:

  1. 你需要支付“社会成本”:如果这个抽象与团队现有的习惯(Lore)相悖——比如在一个从未用过函数式编程的 Go 团队里强推 Monad——你将遭遇巨大的阻力。
  2. 团队的保守性:成熟的团队往往趋于保守,改变既定习惯需要巨大的能量。你不能仅仅因为一个抽象在理论上很美就引入它,你必须证明它的收益足以覆盖它带来的社会摩擦成本
  3. 认知负担是共享的:一个抽象对你来说可能很清晰,但如果它让队友感到困惑,那就是在消耗团队的整体智力资源。

因此,当我们评判一个抽象是否“恰当”时,不能只看代码本身,还必须看它是否“合群”。这正是我们接下来要引入海德格尔哲学的现实基础。

锤子哲学 —— “上手状态” vs. “在手状态”

John 引用了海德格尔在《存在与时间》中的一个著名概念:Ready-to-hand (上手状态)Present-at-hand (在手状态)

  • 上手状态 (Ready-to-hand):当你熟练使用一把锤子钉钉子时,你的注意力完全在钉钉子这件事上,锤子本身在你意识中是“透明”的。你感觉不到它的存在,它只是你身体的延伸。
  • 在手状态 (Present-at-hand):当锤子突然坏了(比如锤头掉了),或者你拿到一把设计奇特的陌生工具时,你的注意力被迫从“钉钉子”转移到了“锤子”本身。你开始审视它的构造、重量和用法。

这对代码意味着什么?

  • 好的抽象是“上手状态”的:比如 for 循环。作为经验丰富的开发者,你使用它时是在思考“我要遍历数据”,而不是“这个循环语法是怎么编译的”。它透明、顺手,让你专注于解决问题。

  • 坏的抽象是“在手状态”的:比如一个复杂的、过度设计的 ORM 或者一个晦涩的 Monad 库。当你使用它时,你的思维被迫中断,你需要停下来思考:“这个函数到底在干什么?这个参数是什么意思?”

如果一个抽象让你频繁地从“解决业务问题”中抽离出来去思考“工具本身”,那么它很可能是一个坏的抽象

注:通过学习和实践,在手状态 (Present-at-hand)的抽象可以转换为 上手状态 (Ready-to-hand)的抽象。

真理的检验 —— “本质真理” vs. “巧合真理”

接着,John 又搬出了康德关于真理的分类,引导我们思考抽象的持久性

  • 分析真理 (Analytic Truth):由定义决定的真理。比如“所有单身汉都没结婚”。在代码中,这就像 unnecessary abstractions are unnecessary,虽然正确但没啥用。
  • 综合真理 (Synthetic Truth):由外部事实决定的真理。比如“外面在下雨”。它的真假取决于环境,随时可能变。
  • 本质真理 (Essential Truth):虽然不是由定义决定,但反映了世界的本质规律。比如“物质由原子构成”。

这对抽象意味着什么?

当你提取一个抽象时,问问自己:它代表的是代码的“本质真理”,还是仅仅是一个“巧合”?

举个例子:你有一段过滤商品的代码,可以按“价格”过滤,也可以按“库存”过滤。你提取了一个 Filter(Product) bool 的抽象。

  • 如果未来所有的过滤需求(如颜色、大小)都能用这个签名解决,那么你发现了一个本质真理。这个抽象是稳固的。
  • 但如果突然来了一个需求:“过滤掉重复的商品”,这个需求需要知道所有商品的状态,而不仅仅是单个商品。原本的 Filter(Product) bool 签名瞬间失效。

如果你提取的抽象仅仅是因为几段代码“长得像”(巧合),而不是因为它们“本质上是一回事”,那么当需求变更时,这个抽象就会崩塌,变成一种负担。

由此可见,好的抽象不是被创造出来的,而是被发现(Recognized)出来的。它们是对代码中某种本质结构的捕捉。

实战指南 —— 如何引入抽象?

最后,John 给出了一个评估抽象是否“恰当”的五步清单:

  1. 明确收益 (Benefit):你到底是为了解决重复、隐藏细节,还是仅仅因为觉得它“很酷”?
  2. 考虑社会成本 (Social Cost):编程是社会活动。这个抽象符合团队的习惯吗?引入它是否需要消耗大量的团队认知成本?(比如在 Go 里强推 Monad等函数式编程的范式)。
  3. 是否处于“上手状态” (Ready-to-hand):它能融入开发者的直觉吗?还是会成为注意力的绊脚石?
  4. 是否本质 (Essential):它是否捕捉到了问题的核心结构,能经得起未来的变化?
  5. 是否涌现 (Emergent):它是你从现有代码中“识别”出来的模式,还是你强加给代码的枷锁?

小结:保持怀疑,但别放弃好奇

Go 社区的“避免不必要的抽象”文化,本质上是对认知负担的防御。我们见过太多为了抽象而抽象的烂代码。但 John 提醒我们,不要因此走向另一个极端——恐惧抽象

正确且必要的抽象是强大的武器,它能让我们驾驭巨大的复杂性。只要我们能像海德格尔审视锤子那样审视我们的代码,区分“上手”与“在手”,区分“本质”与“巧合”,我们就能在 Go 的简约哲学中,找到属于自己的那条“正确”道路。

资料链接:https://www.youtube.com/watch?v=oP_-eHZSaqc


你的“锤子”顺手吗?

用海德格尔的视角审视代码,确实别有一番风味。在你现在的项目中,有哪些抽象是让你感觉“如臂使指”的(上手状态)?又有哪些抽象经常让你
“出戏”,迫使你不得不去研究它内部的构造(在手状态)?

欢迎在评论区分享你的“哲学思考”! 让我们一起寻找那个最本质的代码真理。

如果这篇文章带给你一次思维的“脑暴”,别忘了点个【赞】和【在看】,并转发给那些喜欢深究技术的伙伴!


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

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

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


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

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

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

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

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


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

跨越20年的对话:从 Eiffel 的“契约”到 Go 的“接口”

本文永久链接 – https://tonybai.com/2025/12/13/from-eiffel-contract-to-go-interface

大家好,我是Tony Bai。

20年前,当我第一次翻开 Bertrand Meyer 的那本巨著面向对象软件构造》(Object-Oriented Software Construction) 时,一种醍醐灌顶的感觉油然而生。书中那个名为 Eiffel 的语言,以及它所倡导的 “契约式设计” (Design by Contract, DbC),仿佛为当时混乱的软件开发世界点亮了一盏明灯。

虽然 Eiffel 语言最终并未像 Java 或 C++ 那样统治世界,但它留下的思想遗产——前置条件、后置条件、不变量——却潜移默化地渗透进了现代软件工程的骨髓。

时光流转,当我们站在云原生时代的潮头,手握 Go 语言 这把利器时,你是否意识到:Go 的接口 (Interface) 设计,其实是一场跨越 20 年的、对契约精神的现代演绎与致敬。

今天,让我们重温经典,看看那些曾被奉为圭臬的“契约”,是如何在 Go 的代码世界里重生的。

什么是“契约”?—— 软件世界的商业法则

在人类社会中,商业活动的基石是合同(契约)。甲方(Client)和乙方(Supplier)通过一纸文书,明确了彼此的权利义务

Bertrand Meyer 的天才之处,在于他将这种商业隐喻完美地移植到了软件模块的交互中。他认为,软件的高可靠性不能靠“运气”或“防御性编程的堆砌”,而应靠明确定义的契约

Eiffel 语言直接将这种契约内置到了语法层面,形成了著名的“三驾马车”

  1. 前置条件 (Preconditions / require)

    • 定义:在调用函数之前,调用方 (Client) 必须确保为真的条件。
    • 商业隐喻:你要坐飞机(调用服务),必须先买票且准时到达(满足前置条件)。如果没买票,航空公司(服务方)有权拒绝服务。
  2. 后置条件 (Postconditions / ensure)

    • 定义:在函数执行之后,服务方 (Supplier) 承诺必须为真的条件。
    • 商业隐喻:只要你买了票且准时登机,航空公司必须把你安全送到目的地(满足后置条件)。
  3. 不变量 (Invariants / invariant)

    • 定义:在对象的整个生命周期中(所有公开方法调用前后),始终保持为真的“真理”。
    • 商业隐喻:无论飞机怎么飞,乘客数量绝不能超过座位数。

“契约”的核心价值在于信任:如果每个人都遵守契约,我们就不需要在每一行代码里都写那种偏执的 if (x != null) 检查。代码将变得更干净、更高效、更健壮。

为了让你直观感受这种思想的冲击力,让我们看一段 Eiffel 代码。这是一个简单的字典(Dictionary)插入操作,请注意看它是如何用 require、ensure 和 invariant 将逻辑严丝合缝地包裹起来的:

class DICTIONARY [ELEMENT]

feature
    count: INTEGER
    capacity: INTEGER

    put (x: ELEMENT; key: STRING) is
        -- 将元素 x 插入字典,通过 key 检索
        require
            -- [前置条件]:调用者的责任
            not_full: count < capacity
            key_not_empty: not key.empty
        do
            -- ... 这里是具体的插入算法实现 ...
            -- ... 真正的业务逻辑代码 ...
        ensure
            -- [后置条件]:实现者的承诺
            element_added: has (x)
            key_associated: item (key) = x
            count_increased: count = old count + 1
        end

invariant
    -- [不变量]:始终为真的真理
    consistent_count: 0 <= count and count <= capacity

end

注:对于不熟悉 Eiffel 语法的同学,其实只需关注四个关键词:require 是对入参的“资格审查”,do 是干活的“核心逻辑”,ensure 是对结果的“质量验收”,而 invariant 则是贯穿始终的“宪法”。

看到这里,你是否感受到了一种秩序之美?

这段代码不仅仅是在“写程序”,它是在立法。require 明确了“什么情况下可以调”,ensure 明确了“调用后会发生什么”,而 invariant 则像定海神针一样稳住了对象的状态。

“契约”的核心价值在于信任:如果每个人都遵守契约,我们就不需要在每一行代码里都写那种偏执的 if (x != null) 检查。代码将变得更干净、更高效、更健壮。

Go 接口 —— 契约的“鸭子类型”演绎

Eiffel 选择了显式的、强硬的语法来强制契约;而 Go 语言,则选择了一种更为隐式、灵活,但也更具工程智慧的方式——接口 (Interface)。下面表格直观地展示了在契约这个概念上,Eiffel实现方式与Go的演绎方式上的方式:

下面我们再具体说一下。

行为即契约

Go 的接口设计哲学是:“如果它走起路来像鸭子,叫起来像鸭子,那它就是鸭子。”

在 Go 中,我们不关心一个类型“是谁”(继承了哪个父类),我们只关心它“承诺能做什么”。这种承诺,就是契约。

以标准库中最经典的 io.Reader 为例:

type Reader interface {
    Read(p []byte) (n int, err error)
}

这短短三行代码,实际上定义了一个极其强大的契约:

  • 前置条件(隐式):你需要给我一个切片 p。
  • 后置条件(隐式):我会尝试读取数据填入 p,并返回读取的字节数 n 和可能发生的错误 err。如果 n > 0,则 p[0:n] 包含了有效数据。

任何一个结构体,无论是 os.File、net.Conn 还是 bytes.Buffer,只要它签署(实现)了这个契约,就可以被无缝地替换和复用。这正是 DbC(Design by Contract) 理论中 Liskov 替换原则 在 Go 语言中的完美落地。

强类型的约束

虽然 Go 没有 require 关键字,但它利用强类型系统实施了最基础的契约检查。

在动态语言中,你可能需要写代码检查参数是否为数字。但在 Go 中,如果函数签名是 func Sqrt(x float64),编译器就是你的契约执行官——它保证了绝不会有字符串类型的“非法移民”混入函数内部。

在 Go 中实践“契约精神”

在尝试将 DbC 落地到 Go 语言时,我们必须首先承认一个事实:Go 并非传统的面向对象语言

Eiffel 是建立在类(Class)和继承(Inheritance)之上的。它的 invariant 依赖于类的状态封闭性,它的 require 和 ensure 依赖于方法重写时的“契约继承”规则(Liskov 替换原则的严格形式)。

而 Go 是基于组合接口的。我们没有“类”,只有结构体;我们没有“继承”,只有嵌入。这种范式上的根本差异,注定了我们无法在 Go 中获得 Eiffel 那种“原生级”的契约支持,任何试图在语法层面 1:1 还原 Eiffel 的尝试,都会显得格格不入且笨拙。

但这并不意味着我们可以抛弃 DbC 的思想。相反,一个优秀的 Gopher,应当学会“神似而形不似”——利用 Go 的原生特性(Panic, Error, Defer, Testing),手动“编织”出健壮的契约网。

捍卫前置条件:Panic 还是 Error?

在 Go 中执行前置条件检查,通常有两种流派:

  • 针对编程错误(Bug)—— 使用 panic

如果调用者违反了API的基本使用协议(例如,传入了一个 nil 的上下文,或者索引越界),这通常意味着调用方代码有 Bug。此时,快速失败(Fail Fast)是最好的选择。

func MustRegister(handler Handler) {
    if handler == nil {
        panic("http: nil handler") // 显式的前置条件检查
    }
    // ...
}
  • 针对运行时错误 —— 返回 error

如果前置条件依赖于外部世界(如网络是否连通、文件是否存在),则应返回 error,让调用方决定如何处理。

验证后置条件:Defer 与测试

Eiffel 的 ensure 可以在运行时自动检查。在 Go 中,我们可以利用 defer 甚至构建标签(Build Tags)来模拟这种行为,特别是在调试模式下。

// 仅在调试构建中启用的断言逻辑
func (s *Stack) Push(item int) {
    if debug {
        // 捕获旧状态
        oldSize := s.size
        defer func() {
            // 验证后置条件
            if s.size != oldSize + 1 {
                panic("invariant violated: stack size did not increment")
            }
        }()
    }
    // ... 业务逻辑 ...
}

但更 Go Style 的做法是:将后置条件的验证移交给单元测试(Unit Test)和模糊测试(Fuzzing)。Go 强大的测试工具链,本质上就是一个外挂的“契约验证器”。

守护不变量:“构造函数”与封装

如何保证对象始终处于合法状态(不变量)?Go 给出的答案是:封装(Encapsulation)

通过将结构体的字段设为私有(小写字母开头),并强制用户通过 New… 工厂函数来创建对象,我们可以确保对象在出生那一刻就是满足不变量的,并且在后续的生命周期中,外部无法破坏它。

package stack

type Stack struct {
    items []int // 私有,外部无法直接修改,保证了数据的安全性
}

// 工厂函数:保证初始状态的不变量
func New() *Stack {
    return &Stack{items: make([]int, 0)}
}

示例 —— 一个“契约式”的栈

让我们把上述思想综合起来,写一个简单的、充满“契约精神”的栈。

package stack

import "errors"

// StackInterface 定义了行为契约
type StackInterface interface {
    Push(v int) error
    Pop() (int, error)
    Size() int
}

type Stack struct {
    items []int
    cap   int
}

// New 创建栈,同时确立初始不变量
func New(capacity int) *Stack {
    if capacity <= 0 { // 前置条件检查
        panic("capacity must be positive")
    }
    return &Stack{
        items: make([]int, 0, capacity),
        cap:   capacity,
    }
}

func (s *Stack) Push(v int) error {
    // 前置条件:栈未满
    if len(s.items) >= s.cap {
        return errors.New("stack overflow")
    }

    s.items = append(s.items, v)

    // 后置条件(隐式):len 增加了 1,且栈顶元素是 v
    // 在 Go 中,我们通常信任代码逻辑,或通过测试覆盖此条件
    return nil
}

func (s *Stack) Pop() (int, error) {
    // 前置条件:栈不为空
    if len(s.items) == 0 {
        return 0, errors.New("stack underflow")
    }

    v := s.items[len(s.items)-1]
    s.items = s.items[:len(s.items)-1]
    return v, nil
}

// 不变量:Size 永远不会超过 Capacity,也不会小于 0
// 这由 Push 和 Pop 的逻辑严密性以及私有字段的封装共同保证。

进阶思考:并发下的不变量

还有一点不能忽略:Go 是为并发而生的。在单线程模型中,封装或许足以维护不变量。但在 Go 的并发世界里,如果多个 goroutine 同时修改这个 Stack,竞态条件(Race Condition)瞬间就会破坏 count <= capacity 这样的“真理”。

因此,在 Go 的工程实践中,维护不变量往往还需要同步原语(如 sync.Mutex)的强力介入。只有配合了锁机制,才能确保对象在并发洪流的冲击下,依然能守住那份“不变”的契约。

小结:心中的契约

在结束这次跨越 20 年的时空对话之际,我想特别澄清一点:本文的目的,绝非鼓励大家在 Go 语言中笨拙地“模拟”一套 Eiffel 的语法糖。

Go 语言有其独特且自洽的设计哲学——简洁、组合、并发。强行在 Go 代码中堆砌 require() 或 ensure() 函数,往往会画虎不成反类犬,破坏 Go 代码原有的流畅性。

我们重温 DbC,是为了汲取思想的养分。Bertrand Meyer 教会了我们要对代码的“权利与义务”保持敏感:

  • 当你写下一个函数时,你是否想清楚了它的前置条件
  • 你是否通过单元测试守护了它的后置条件
  • 你是否通过封装维护了对象的不变量

这些思考方式,才是 DbC 留给非 DbC 语言(如 Go、Java、Python)最宝贵的遗产。Bertrand Meyer 在 20 年前种下的那颗种子,虽然没有长成 Eiffel 这棵参天大树,但它的花粉却飘散到了整个软件工程的花园里。

Go 语言选择了另一条更务实的道路:用接口定义契约,用封装保护契约,用测试验证契约。

作为一名 Gopher,当我们写下 type … interface,或者敲下 if err != nil 时,我们实际上是在履行一份神圣的职责。语言的特性在演进,但软件工程的核心——信任与责任的管理——从未改变。

真正的契约,不只写在代码里,更应刻在每一位工程师的心里。

参考资料

  • Building bug-free O-O software: An introduction to Design by Contract – https://archive.eiffel.com/doc/manuals/technology/contract/
  • Object-Oriented Software Construction(2nd) – https://book.douban.com/subject/1547078/
  • Programming “By Contract” – https://www.cs.usfca.edu/~parrt/course/601/lectures/programming.by.contract.html

聊聊你心中的“代码契约”

这场跨越20年的思想对话,让我们重新审视了Go接口背后那份深刻的工程哲学。从Eiffel那严谨如“立法”的require/ensure,到Go语言“润物细无声”的interface/error/testing组合,我们看到的是不同时代背景下,对“信任与责任”这一软件工程核心母题的不同解答。

那么,在你日常的Go编程实践中,你是如何理解和贯彻“契约精神”的?

  • 你是否也有过因为接口(契约)定义不清,而导致团队协作“踩坑”的经历?
  • 除了文中提到的方法,你还有哪些维护代码“权利与义务”的独门心法?
  • 你认为,Go语言在“契约”的表达上,还有哪些值得改进或探索的方向?

非常期待在评论区看到你的故事与真知灼见,让我们一起探讨如何成为更具“契约精神”的工程师!

如果这篇文章让你对Go接口或软件工程的理解更深了一层,别忘了点个【赞】和【在看】,并分享给更多热爱思考的同伴!


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

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

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


你的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