bug|「Hello World」中的「bug」
选自sunfishcode博客
作者:sunfishcode
机器之心编译
机器之心编辑部
Hello World 可能是许多人编写的第一个程序 。 这么简单的程序按理说应该没有 bug 吧?一位叫「sunfishcode」的开发者给出了令人意外的结论 。C 语言中的 Hello World
用 C 语言写 Hello World 有很多种不同的方式 , 比如维基百科里记录的版本、K&R book 中介绍的版本 , 甚至还有 1974 年的原始版本 。

文章图片
这里展示一个 ANSI C 的版本:
/* Hello World in C, Ansi-style */
#include <stdio.h>
#include <stdlib.h>
int main(void)
【bug|「Hello World」中的「bug」】{
puts("Hello World!");
return EXIT_SUCCESS;}
这个版本使用 (void) 来确保 main 是一个新型的声明 。 它使用 EXIT_SUCCESS 宏 , 而不是假设平台使用 0 表示 success , 根据 C 的标准 , 这是不必要的 。 但我们在这里不会冒任何风险 。 它使用适当的头文件以避免隐式声明 puts 。
这个版本试图把每件事都做好 , 但它仍然有一个缺陷 。
上面提到的所有版本都有一个 bug 。
bug 在哪儿?
Linux 有一个有趣的设备文件 , 叫做「/dev/full」 , 就像它更著名的表亲「/dev/null」一样 。 但是当你写入「/dev/full」时 , 它不会丢弃数据 , 而是会失败 。 它的作用就像文件系统中一个刚刚耗尽空间的文件:
$ echo "Hello World!" > /dev/full
bash: echo: write error: No space left on device
$ echo $?
1
这是一个很好的小工具 , 用于测试程序能否正确处理 I/O 错误 。 如果没有剩余的空间 , 或者磁盘出现故障 , 那么创建实际的文件系统是很不方便的 , 但是让一个程序将其输出写入「/dev/full」 , 然后看看会发生什么 , 这是非常容易的 。
让我们测试一下上面的 C 语言例子:
$ gcc hello.c -o hello$
./hello > /dev/full$
echo $?
0
与在上面的 shell 中使用 echo 不同 , 这里没有输出 , 退出状态为零 。 这意味着 hello 程序报告了成功执行 。 然而 , 它实际上并没有成功 。 我们可以通过使用 strace 确认它遇到了故障 。
$ strace -etrace=write ./hello > /dev/full
write(1, "Hello World!\n", 13) = -1 ENOSPC (No space left on device)
+++ exited with 0 +++
操作系统报告了「No space」错误 , 但没关系 , 程序默默地接受它并返回 0 , 这是成功的代码 。 这是一个 bug!
这个 bug 有多严重?可以说 , hello world 在任何地方都不会是安全的 。 然而 , hello world 确实做了一些现实世界的程序所做的事情:打印到标准输出 , 这可能会被重定向到一个文件 。 在现实世界中 , 文件可能会耗尽空间 。 如果一个程序没有检测到这种错误并通过其返回代码报告该错误 , 那么它的父进程将不知道子进程失败了 , 并且将继续运行 , 就像没有任何错误一样 , 即使它期望产生的输出已经悄悄地丢失了数据 。
例如 , 考虑一个将 yaml 文件打印到标准输出的程序 。 如果标准输出耗尽空间 , 则输出可能会在某个任意点被截断 , 尽管它可能仍然是有效的 yaml 。 因此 , 我们应该期待程序能够检测和报告这种情况 。
如果换成其他语言呢?
在前面的内容中 , 我们重点看了 bash 和 C , 那如果换成 Python 呢?Python 处理错误的原则可是「Errors should never pass silently」 。 以下是 Python 2 的情况:
$ python2 hello.py > /dev/full
close failed in file object destructor:
sys.excepthook is missing
lost sys.stderr
$ echo $?
0
它确实向 stderr 打印了一条消息 , 尽管这是一条令人困惑的消息 。 但是 , 它也返回 0 , 这意味着它告诉运行它的人它已经成功退出 。
幸运的是 , Python 3 正确地报告了错误 , 并打印了一个更好的错误消息:
$ python3 hello.py > /dev/full
Exception ignored in: <_io.TextIOWrapper name='<stdout>' mode='w' encoding='utf-8'>
OSError: [Errno 28] No space left on device
$ echo $?
120
最后 , 作者又尝试了几种语言 , 结果如下:

文章图片

文章图片
原文链接:https://blog.sunfishcode.online/bugs-in-hello-world/
封面出自:https://lmichelin.fr/hello-world/
推荐阅读
- 手机|谁是2000元「平民机皇」?天玑8100双雄优劣对比,看完不买错...
- 操作|Mac 的内置硬盘速度那么快,它是不是「作弊」了|免费试读
- 苹果|15 年后,苹果重新变回「电脑公司」
- 系列|助推武汉产业发展,「创投W+」系列项目路演精彩来袭
- 扫描|花费不到 800 元,我把家里的老照片全部变成了「高清数字版」
- 罗永浩|极客简报|高端 iPhone 专享「叹号屏」;罗永浩泄露新公司命名;华为把稻草卖出金子价
- 平板|华为 MatePad Paper 体验:MatePad & 电子书「跨界选手」
- 网络|越「丑」越快乐,这就是年轻人新的社交大法
- 中芯国际|滴滴6月或发布造车计划;英特尔顶级专家Mike Burrows跳槽AMD;Android 13开发者预览版2发布|极客头条
- 王者|这个曾经的国产HIFI耳机王者,竟然还活着?