引用计数为何成为现代语言的隐形选择:从性能开销到优化突围

1960年,John McCarthy在MIT实现第一个LISP解释器时,面临一个看似简单却影响深远的问题:如何在程序运行时自动回收不再使用的内存?他给出的答案是tracing garbage collection——周期性地遍历所有可达对象,标记并清理不可达的内存。这个方案统治了函数式语言数十年。 ...

12 min · 5977 words

类型系统的五十年设计博弈:为什么每个选择都意味着放弃

1960年代,Bertrand Russell为了解决集合论悖论而提出类型理论时,大概没想到这个概念会在六十年后成为程序员日常争论的焦点。今天,每一个新编程语言的诞生都伴随着同样的灵魂拷问:静态类型还是动态类型?结构化还是名义?类型擦除还是单态化? ...

11 min · 5482 words

错误处理为何分裂五十年:从PL/I的ON语句到Rust的Result类型

1964年,IBM在设计PL/I语言时引入了一个被称为"ON语句"的构造。这个看似简单的语法元素,开创了编程语言中结构化错误处理的先河。六十年后,当Rust的?操作符和Go的显式错误检查成为现代语言的主流选择时,我们不禁要问:为什么错误处理这个问题困扰了语言设计者如此之久? ...

19 min · 9052 words

词法作用域与动态作用域:为什么这个六十年前的设计抉择至今仍在影响你的代码

1975年,麻省理工学院的 Gerald Sussman 和 Guy Steele 正在开发一门新的 Lisp 方言。他们遇到了一个看似简单的问题:当一个函数引用外部变量时,它应该找到哪个值?这个问题困扰了编程语言设计者十五年,而他们的解决方案——采用词法作用域——后来成为现代编程语言的标配。 ...

15 min · 7062 words

泛型的三种命运:为什么同一个概念在不同语言中走向了截然不同的实现道路

一个概念,三种命运 1975年,Robin Milner在ML语言中首次引入了参数化多态(parametric polymorphism)——这便是我们今天所称"泛型"的理论源头。半个世纪过去,这个概念已经渗透进几乎所有主流编程语言。然而,当你打开C++、Java、Go、Rust、C#的编译器源码,会发现同一个"泛型"概念在这些语言中竟然演化出了截然不同的实现策略。 ...

16 min · 7926 words

类型推断的错误消息为何如此难懂:从统一算法到错误定位的四十年迷局

一位TypeScript开发者在重构代码时遇到了这样一个错误: Type 'string' is not assignable to type 'number'. 错误指向了一个看似完全正确的函数调用。他检查了函数签名,检查了传入的参数类型,甚至检查了调用链上的每一个函数,花了整整两个小时才找到问题的根源——一个在文件另一端、三百行之外的对象属性赋值。 ...

9 min · 4270 words

代码注释的消亡——为什么现代代码越来越不需要注释

1984年,Donald Knuth在斯坦福大学提出了一个激进的设想:程序员应该像散文家一样写作。在他著名的论文《Literate Programming》中,Knuth写道:“让我们改变传统的编程观念:不再想象我们的主要任务是指示计算机做什么,而是专注于向人类解释我们想让计算机做什么。” ...

10 min · 4958 words