编程匠艺——编写卓越的代码
分類: 图书,计算机/网络,程序设计,其他,
作者: (美)古德利弗著,韩江,陈玉译
出 版 社: 电子工业出版社
出版时间: 2008-9-1字数: 840000版次: 1页数: 582印刷时间: 2008/09/01开本: 16开印次: 1纸张: 胶版纸I S B N : 9787121069802包装: 平装编辑推荐
《程序员》杂志技术主编孟岩作序推荐!作者Pete Goodliffe是美国资深软件开发专家!助你在现实世界重重困难的情况下编写出优秀的代码!
本书主要内容:
《编程匠艺》中随处可见与编程语言无关、但是与所有开发人员都相关的建议,其内容遍及编程的各个要素,如代码风格、变量命名、错误处理和安全性等。不仅如此,它还触及了一些更广泛的现实世界编程问题,如有效的团队合作、开发过程和文档编写,等等。本书各章末尾均有一个问题和答案部分,这些部分回顾了各章中的一些关键概念,促使你像专家一样思考,从而使这本书成为那些渴望作为团队的一分子、专业并高效地编程的新手们的一本绝佳的参考书。本书将告诉你如何:
在现实世界困难重重的情况下编写出优秀的代码;
避免在工作场所出现灾难和人心涣散的情况;
准确地评估你的能力,并找到提高的途径;
采取有建设性的态度,并遵从最佳的做法。
内容简介
如果你可以编写出合格的代码,但是想更进一步、创作出组织良好而且易于理解的代码,并希望成为一名真正的编程专家或提高现有的职业技能,那么《编程匠艺——编写卓越的代码》都会为你给出答案。本书的内容遍及编程的各个要素,如代码风格、变量命名、错误处理和安全性等。此外,本书还对一些更广泛的编程问题进行了探讨,如有效的团队合作、开发过程和文档编写,等等。本书各章的末尾均提供一些思考问题,这些问题回顾了各章中的一些关键概念,可以促使你像专家一样思考,从而使本书成为那些渴望作为团队的一分子,职业并高效地编程的新手们的一本绝佳的参考书。
作者简介
Pete Goodliffe是一位软件开发专家,他在软件“食物链”上从未驻足不前。他在各种各样的项目中使用过许多种语言。他还在教授和指导程序员方面有着丰富的经验,并且常年为ACCU的C Vu杂志(www.accu.org)撰写栏目“编程的职业化”。Pete痴迷于编写出色的、没有错误的代码,这使得他有更多的时间与自己的孩子共
目录
第Ⅰ篇 代码表面第一部分
第1章 善于防守——健壮代码的防御性编程技巧
1.1 向优秀的代码前进
1.2 设想:最坏的选择
1.3 什么是防御性编程
1.4 又大又坏的世界
1.5 防御性编程技巧
1.5.1 使用好的编码风格和合理的设计
1.5.2 不要仓促地编写代码
1.5.3 不要相信任何人
1.5.4 编码的目标是清晰,而不是简洁
1.5.5 不要让任何人做他们不该做的修补工作
1.5.6 编译时打开所有警告开关
1.5.7 使用静态分析工具
1.5.8 使用安全的数据结构
1.5.9 检查所有的返回值
1.5.10 审慎地处理内存(和其他宝贵的资源)
1.5.11 在声明位置初始化所有变量
1.5.12 尽可能推迟一些声明变量
1.5.13 使用标准语言工具
1.5.14 使用好的诊断信息日志工具
1.5.15 审慎地进行强制转换
1.5.16 细则
1.6 约束
1.6.1 约束的内容
1.6.2 移除约束
1.7 总结
1.8 另请参见
1.9 思考
1.9.1 深入思考
1.9.2 结合自己
第2章 精心布局——源代码的版面和样式
2.1 什么是关键
2.2 了解你的读者
2.3 什么是好的样式
2.4 使用括号
2.4.1 K&R括号风格
2.4.2 悬挂式的括号风格
2.4.3 缩进的括号风格
2.4.4 其他的括号风格
2.5 主宰一切的风格
2.6 内部风格(以及在哪里使用它们)
2.7 设立标准
2.8 正义的战争
2.9 总结
2.10 另请参见
2.11 思考
2.11.1 深入思考
2.11.2 结合自己
第3章 名正言顺——为有意义的事物起有意义的名称
3.1 为什么我们应该恰当地命名呢
3.2 我们对什么进行命名
3.3 名字游戏
3.3.1 描述性
3.3.2 技术上正确
3.3.3 符合语言习惯
3.3.4 恰当
3.4 具体细节
3.4.1 命名变量
3.4.2 命名函数
3.4.3 命名类型
3.4.4 命名名字空间
3.4.5 命名宏
3.4.6 命名文件
3.5 玫瑰不叫玫瑰
3.5.1 保持前后一致
3.5.2 利用上下文
3.5.3 使用对你有利的名称
3.6 总结
3.7 另请参见
3.8 思考
3.8.1 深入思考
3.8.2 结合自己
第4章 不言自明——编写“自文档化”代码的技巧
4.1 自文档化的代码
4.2 编写自文档化代码的技术
4.2.1 使用好的样式编写简单的代码
4.2.2 选择有意义的名称
4.2.3 分解为原子函数
4.2.4 选择描述性的类型
4.2.5 命名常量
4.2.6 强调重要的代码
4.2.7 分组相关信息
4.2.8 提供文件头
4.2.9 恰当地处理错误
4.2.10 编写有意义的注释
4.3 实用的自文档化方法
4.3.1 文学性编程
4.3.2 文档化工具
4.4 总结
4.5 另请参见
4.6 思考
4.6.1 深入思考
4.6.2 结合自己
第5章 随篇注释——如何编写代码注释
5.1 什么是代码注释
5.2 注释看上去是什么样的
5.3 多少注释是恰当的
5.4 注释中应该有些什么
5.4.1 解释为什么,而不是怎么样
5.4.2 不要描述代码
5.4.3 不要取代代码
5.4.4 确保注释有用
5.4.5 避免分心
5.5 实践
5.6 从审美的角度看注释
5.6.1 一致性
5.6.2 清晰的块注释
5.6.3 缩进的注释
5.6.4 行尾注释
5.6.5 帮助你阅读代码
5.6.6 选择一种维护成本较低的风格
5.6.7 分隔板
5.6.8 标志
5.6.9 文件头注释
5.7 使用注释
5.7.1 帮助你编写例行程序
5.7.2 错误修正通告
5.7.3 注释过时
5.7.4 维护和空洞无物的注释
5.8 总结
5.9 另请参见
5.10 思考
5.10.1 深入思考
5.10.2 结合自己
第6章 人非圣贤——处理不可避免的情况——代码中的错误情形
6.1 从何而来
6.2 错误报告机制
6.2.1 不报告
6.2.2 返回值
6.2.3 错误状态变量
6.2.4 异常
6.2.5 信号
6.3 检测错误
6.4 处理错误
6.4.1 何时处理错误
6.4.2 可能的反应
6.4.3 代码示例
6.5 使地狱浮现
6.6 管理错误
6.7 总结
6.8 另请参见
6.9 思考
6.9.1 深入思考
6.9.2 结合自己
第Ⅱ篇 代码的神秘生命
第7章 欲善其事,先利其器——使用工具构建软件
7.1 什么是软件工具
7.2 为什么要在意工具
7.3 使工具发挥作用
7.3.1 了解它能做些什么
7.3.2 学习如何驾驭它
7.3.3 了解它适合什么任务
7.3.4 检查它是否可用
7.3.5 找到了解更多信息的途径
7.3.6 查明新版本何时出现
7.4 哪个工具
7.4.1 源代码编辑工具
7.4.2 代码构建工具
7.4.3 调试和调查工具
7.4.4 语言支持工具
7.4.5 其他工具
7.5 总结
7.6 另请参见
7.7 思考
7.7.1 深入思考
7.7.2 结合自己
第8章 测试时代——测试代码的魔术
8.1 反思现实
8.2 谁、是什么、何时以及为什么
8.2.1 我们为什么要测试
8.2.2 谁来进行测试
8.2.3 测试的内容有些什么
8.2.4 何时进行测试
8.3 测试并不难……
8.4 测试的类型
8.5 选择单元测试用例
8.6 为测试而设计
8.7 看!不要用手!
8.8 面对故障该怎么办
8.9 你能管理它吗
8.9.1 缺陷跟踪系统
8.9.2 bug审查
8.10 总结
8.11 另请参见
8.12 思考
8.12.1 深入思考
8.12.2 结合自己
第9章 寻找缺陷——调试:当事情进展得不顺利时该怎么办
9.1 生活的真相
9.2 bug的种类
9.2.1 从远处看
9.2.2 从近处看
9.2.3 从更近处看
9.3 消灭害虫
9.3.1 地下之路
9.3.2 地上之路
9.4 搜寻bug
9.4.1 编译时错误
9.4.2 运行时错误
9.5 如何修正缺陷
9.6 预防
9.7 除蜂剂、驱虫剂、捕蝇纸
9.7.1 调试器
9.7.2 内存访问校验器
9.7.3 系统调用跟踪
9.7.4 内核转储
9.7.5 日志
9.8 总结
9.9 另请参见
9.10 思考
9.10.1 深入思考
9.10.2 结合自己
第10章 代码构建——将源代码转换为可执行代码的过程
10.1 语言障碍
10.1.1 解释型语言
10.1.2 编译型语言
10.1.3 字节编译型语言
10.2 小题大做
10.3 构建软件版本
10.4 怎样才算是一个优秀的构建系统
10.4.1 简洁
10.4.2 一致
10.4.3 可重复和可靠
10.4.4 原子性
10.4.5 能够应付错误
10.5 技术细节
10.5.1 目标的选择
10.5.2 内务处理
10.5.3 依赖关系
10.5.4 自动构建
10.5.5 构建配置
10.5.6 递归地使用make
10.6 请发布我吧
10.7 构建大师是全能的吗
10.8 总结
10.9 另请参见
10.10 思考
10.10.1 深入思考
10.10.2 结合自己
第11章 追求速度——优化程序和编写高效的代码
11.1 优化是什么
11.2 是什么使代码不尽如人意
11.3 为什么不进行优化呢
备选方案
11.4 为什么要进行优化
11.5 优化的具体细节
11.5.1 证明你需要进行优化
11.5.2 找出运行得最慢的代码
11.5.3 测试代码
11.5.4 优化代码
11.5.5 优化之后
11.6 优化的技术
11.6.1 设计更改
11.6.2 代码更改
11.7 编写高效的代码
11.8 总结
11.9 另请参见
11.10 思考
11.10.1 深入思考
11.10.2 结合自己
第12章 不安全感综合症——编写安全的程序
12.1 危险
12.2 敌人
12.3 借口,都是借口
12.4 感到很脆弱
12.4.1 不安全的设计和体系结构
12.4.2 缓冲溢出
12.4.3 嵌入的查询字符串
12.4.4 竞争状况
12.4.5 整数溢出
12.5 防范措施
12.5.1 系统安装技术
12.5.2 软件设计技术
12.5.3 代码实现技术
12.5.4 规程技术
12.6 总结
12.7 另请参见
12.8 思考
12.8.1 深入思考
12.8.2 结合自己
第Ⅲ篇 代码的形成过程
第13章 崇尚设计——如何创作出优秀的软件设计
13.1 边设计边编程
13.2 我们要设计什么
13.3 为什么这么忙乱
13.4 良好的软件设计
13.4.1 简洁
13.4.2 优雅
13.4.3 模块化
13.4.4 良好的接口
13.4.5 可扩展性
13.4.6 避免重复
13.4.7 可移植性
13.4.8 符合语言习惯
13.4.9 良好地文档化
13.5 如何设计代码
13.5.1 设计方法和过程
13.5.2 设计工具
13.6 总结
13.7 另请参见
13.8 思考
13.8.1 深入思考
13.8.2 结合自己
第14章 软件体系结构——奠定软件设计的基础
14.1 什么是软件体系结构
14.1.1 软件蓝图
14.1.2 视图
14.1.3 在何时和何处进行体系结构设计
14.1.4 用体系结构来做什么
14.1.5 关于组件和连接
14.2 什么是良好的体系结构
14.3 体系结构风格
14.3.1 没有体系结构
14.3.2 分层的体系结构
14.3.3 管道和过滤器体系结构
14.3.4 客户端/服务器体系结构
14.3.5 基于组件的体系结构
14.3.6 框架
14.4 总结
14.5 另请参见
14.6 思考
14.6.1 深入思考
14.6.2 结合自己
第15章 改良与革命——代码是如何成长的
15.1 软件腐烂
15.2 警告信号
15.3 代码是如何成长的
15.4 相信不可能之事
15.5 对此我们可以做些什么
15.5.1 编写新代码
15.5.2 维护现有代码
15.6 总结
15.7 另请参见
15.8 思考
15.8.1 深入思考
15.8.2 结合自己
第Ⅳ篇 “一群”程序员第一部分
第16章 代码猴子——培养正确的编程态度和方法
16.1 各种各样的猴子
16.1.1 卖力工作的程序员
16.1.2 代码猴子
16.1.3 权威
16.1.4 半权威
16.1.5 傲慢的天才
16.1.6 牛仔
16.1.7 规划者
16.1.8 老前辈
16.1.9 狂热者
16.1.10 单线条程序员
16.1.11 拖沓者
16.1.12 勉强的团队领导
16.1.13 你
16.2 理想的程序员
16.3 那么该怎么办
16.4 最愚蠢的人
16.5 总结
16.6 另请参见
16.7 行为表格
16.8 思考
16.8.1 深入思考
16.8.2 结合自己
第17章 团结就是力量——团队合作与个人程序员
17.1 我们的团队——概览
17.2 团队组织
17.2.1 管理方法
17.2.2 责任划分
17.2.3 组织和代码结构
17.3 团队合作工具
17.4 团队疾病
17.4.1 巴别塔
17.4.2 独裁制
17.4.3 民主制
17.4.4 卫星站
17.4.5 大峡谷
17.4.6 流沙
17.4.7 旅鼠
17.5 良好团队合作的个人技巧和特点
17.5.1 沟通
17.5.2 谦虚
17.5.3 处理冲突
17.5.4 学习和适应能力
17.5.5 了解你的不足之处
17.6 团队合作原则
17.6.1 集体代码所有制
17.6.2 尊重别人的代码
17.6.3 编码准则
17.6.4 定义成功
17.6.5 定义责任
17.6.6 避免倦怠
17.7 团队的生命周期
17.7.1 团队的创建
17.7.2 团队的成长
17.7.3 团队合作
17.7.4 团队结束
17.8 总结
17.9 另请参见
17.10 行为表格
17.11 思考
17.11.1 深入思考
17.11.2 结合自己
第18章 安全措施——源代码控制与自我控制
18.1 我们的责任
18.2 源代码控制
18.2.1 修订控制
18.2.2 访问控制
18.2.3 处理代码库
18.2.4 在代码树上创建分支
18.2.5 源代码控制简史
18.3 配置管理
18.4 备份
18.5 发布源代码
18.6 应该将源代码放在哪里
18.7 总结
18.8 另请参见
18.9 思考
18.9.1 深入思考
18.9.2 结合自己
第Ⅴ篇 开发过程的组成部分第一部分
第19章 注意细节——编写软件规范
19.1 规范到底是什么
19.2 规范的类型
19.2.1 需求规范
19.2.2 功能规范
19.2.3 系统体系结构规范
19.2.4 用户界面规范
19.2.5 设计规范
19.2.6 测试规范
19.3 规范应当包含哪些内容
19.4 规范编写过程
19.5 我们为什么会不编写规范
19.6 总结
19.7 另请参见
19.8 思考
19.8.1 深入思考
19.8.2 结合自己
第20章 代码审查——执行代码审查
20.1 什么是代码审查
20.2 何时进行审查
20.2.1 是否要进行审查
20.2.2 审查哪些代码
20.3 执行代码审查
20.3.1 代码审查会议
20.3.2 集成审查
20.4 审查你的态度
20.4.1 作者的态度
20.4.2 审查人员的态度
20.5 完美的代码
20.6 代码审查之外
20.7 总结
20.8 另请参见
20.9 清单
20.10 思考
20.10.1 深入思考
20.10.2 结合自己
第21章 时间估计——软件时间范围估计的魔术
21.1 在黑暗中摸索
21.2 为什么估计这么困难?
21.3 压力之下
21.4 实用的估计方法
21.5 计划游戏
21.6 坚持!
21.7 总结
21.8 另请参见
21.9 思考
21.9.1 深入思考
21.9.2 结合自己
第Ⅵ篇 从高处鸟瞰
第22章 程序秘方——代码开发的方法和过程
22.1 编程风格
22.1.1 结构化编程
22.1.2 面向对象的程序设计
22.1.3 函数式编程
22.1.4 逻辑编程
22.2 烹饪方法:做什么与怎样做
22.3 开发过程
22.3.1 混乱
22.3.2 瀑布模型
22.2.3 SSADM和PRINCE
22.3.4 V模型
22.3.5 原型设计
22.3.6 迭代和增量开发
22.3.7 螺旋模型
22.3.8 敏捷的方法
22.3.9 其他开发过程
22.4 已经够了!
22.5 选择一种过程
22.6 总结
22.7 另请参见
22.8 思考
22.8.1 深入思考
22.8.2 结合自己
第23章 编程领域大观——不同的编程分支
23.1 应用程序编程
23.1.1 塑装软件
23.1.2 定制应用程序
23.2 游戏编程
23.3 系统编程
23.4 嵌入式编程
23.5 分布式编程
23.6 网络应用程序编程
23.7 企业编程
23.8 数字编程
23.9 那又怎样
23.10 总结
23.11 另请参见
23.12 思考
23.12.1 深入思考
23.12.2 结合自己
第24章 下一步呢——结果好就一切都好
但下一步该做什么呢?
答案和讨论
参考书目
索引
媒体评论
“有些书你不得不读,有些书你必须去读。Pete的书就属于后者——它不仅非常有用,而且十分有趣,能够让你成为一名更加优秀的程序员。”
——Jez Higgins,ACCU主席
Pete Goodliffe在业界的年头快要超过好多人的年龄了,此君曾经涉猎多个领域、不同的编程语言以及多种架构,并且曾经在采用不同流程的公司里从事过开发工作。在本书中,他把多年压箱底的一些观念想法和技巧告诉了大家,这些都是时间与智慧的结合,相信无论是开发人员、项目经理甚至测试人员,都可以从中发现阿里巴巴开启金库的钥匙。
——译者
书摘插图
第1章 善于防守——健壮代码的防御性编程技巧
1.1 向优秀的代码前进
似乎可用的代码、正确的代码和优秀的代码,这三种代码之间存在着巨大的差别。M.A. Jackson曾写道:“软件工程师的智慧,就在于他们是否开始意识到:使程序能用和使程序正确,这两者之间有什么样的差别。”(见参考书目Jackson 75)这个差别就在于:
编写在大多数情况下都能用的代码是很容易的。你只要提供常规的输入集,这些代码就会给出常规的输出集。但是,如果提供一些意外的输入,这些代码可能就会崩溃。
正确的代码是绝不会崩溃的。对于所有可能的输入集,它的输出都将是正确的。不过,所有可能输入的集合常常大得惊人,并且难以测试。
然而,并非所有正确的代码都是优秀的代码,因为有些正确代码的逻辑可能很难理解,其代码可能很不自然,并且可能几乎无法维护。
由此可见,编写优秀的代码才是我们应该不懈追求的目标。优秀的代码是健壮的、高效的,当然也是正确的。当面对不常见的输入时,产品级的代码不会崩溃,也不会产生错误的结果。它同时还满足所有其他的要求,包括线程安全、时间约束和重入等。
在自己舒适的家中编写优秀的代码是一回事。而在软件工厂的压力下编写优秀的代码则完全是另外一回事,在那里,你周围的世界在不断地变化,代码库在飞速地发展,而且你还会经常面对风格奇异的“遗留代码”——那些由现在早已不在的代码猴子所编写的旧程序。当这个世界百般阻挠你的编程事业的时候,为写出优秀的代码而努力吧!
在这种充满困难的环境中,怎样才能保证所编写的代码稳定耐用呢?答案是“防御性编程”。
编写代码的方式多种多样(如面向对象的方式、基于组件的模型、结构化设计以及极限编程,等等),但防御性编程却是一种普遍适用的方法。防御性编程与其说是一种正式的方法论,还不如说是一些非正式的基本指导原则。它虽然不是包治百病的万能药,但却是一种预防代码漏洞百出的有效手段。
……