Cleaner, more elegant, and harder to recognize(翻译)
Cleaner, more elegant, and harder to recognize
更整洁,更优雅,但更难识别
看来,有些人把我几个月前一篇文章的标题“Cleaner,more elegant,and wrong”解释成了对异常通用处理的引用。(参见参考文献【35】,注意到电文甚至为我改变了文章的标题)
该文章的标题是对我从一本书中复制的特定代码段的引用,该书的作者声称他提供的代码“更整洁,更优雅”。我指出 ,代码片段不仅更更整洁,更优雅,也是错误的。
你可以编写正确的基于异常的程序
你要记住,这很难。
另一方面,虽然一些事情是很难的,但并不意味着不应该做。
下面是细节:
很容易 难 很难
写不好的基于错误码的代码 写好的基于错误码的代码 写好的基于异常的代码
写不好的基于异常的代码
编写差的代码很容易,无论用什么错误模型
编写好的基于错误码的代码比较难,因为你必须检查每个错误码,并考虑发生错误时应该如何做。
编写好的基于异常的代码非常难,因为你你必须检查每一行代码(实际上是每个子表达式),并考虑它可能引发什么异常以及代码对异常如何进行反应。(在C++中,这并不是那么糟糕,因为C++异常只在执行过程中的特定点抛出,在C#中,可以随时抛出异常)
但是没有关系,就像我说的那样,虽然一些事情很难但不意味着不应该做。写一个设备驱动程序很难,但是人们做了,这是一件好事情。
下面是另外一个表
|
很容易 |
难 |
很难 |
|
认识到基于错误码的代码写得差劲 |
认识到基于错误码的代码写得不差 |
认识到基于异常的代码写得差 |
|
认识到差的基于错误码的代码和不差的 基于错误码的代码之间的差别 |
认识到基于异常的代码写得不差 |
|
|
认识到差的基于异常的代码与不差的基于异常的代码间的差别 |
||
下面是一些虚构的基于错误码的代码。看看你把它划分为“差”或“不差”
BOOL ComputeChecksum(LPCTSTR pszFile, DWORD* pdwResult)
{
HANDLE h = CreateFile(pszFile, GENERIC_READ, FILE_SHARE_READ,
NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
HANDLE hfm = CreateFileMapping(h, NULL, PAGE_READ, 0, 0, NULL);
void *pv = MapViewOfFile(hfm, FILE_MAP_READ, 0, 0, 0);
DWORD dwHeaderSum;
CheckSumMappedFile(pvBase, GetFileSize(h, NULL),
&dwHeaderSum, pdwResult);
UnmapViewOfFile(pv);
CloseHandle(hfm);
CloseHandle(h);
return TRUE;
}
这段代码显然不好。没有检查错误码。这是你匆匆忙忙时可能写的那种代码,意思是稍后再来修改。而且很容易发现,这个代码在成为好代码前需要进行大量的改进。
下面是另外一个版本:
BOOL ComputeChecksum(LPCTSTR pszFile, DWORD* pdwResult)
{
BOOL fRc = FALSE;
HANDLE h = CreateFile(pszFile, GENERIC_READ, FILE_SHARE_READ,
NULL, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL);
if (h != INVALID_HANDLE_VALUE) {
HANDLE hfm = CreateFileMapping(h, NULL, PAGE_READ, 0, 0, NULL);
if (hfm) {
void *pv = MapViewOfFile(hfm, FILE_MAP_READ, 0, 0, 0);
if (pv) {
DWORD dwHeaderSum;
if (CheckSumMappedFile(pvBase, GetFileSize(h, NULL),
&dwHeaderSum, pdwResult)) {
fRc = TRUE;
}
UnmapViewOfFile(pv);
}
CloseHandle(hfm);
}
CloseHandle(h);
}
return fRc;
}
这段代码仍然是错误的,但它看起来像是试图达到正确。这正是我说的“不差”。
下面是一些基于异常的代码,你可能在匆忙中写下:
NotifyIcon CreateNotifyIcon()
{
NotifyIcon icon = new NotifyIcon();
icon.Text = "Blah blah blah";
icon.Visible = true;
icon.Icon = new Icon(GetType(), "cool.ico");
return icon;
}
(这是一段关于任务栏通知图标的文章中摘录的真实程序的实际代码。轻微的变化是掩盖来源的徒劳无益的尝试)
下面的代码像是你修改之后的样子,能够在异常情况下正确处理,
NotifyIcon CreateNotifyIcon()
{
NotifyIcon icon = new NotifyIcon();
icon.Text = "Blah blah blah";
icon.Icon = new Icon(GetType(), "cool.ico");
icon.Visible = true;
return icon;
}
微妙的,不是吗?
很容易发现差的基于错误码的代码和不差的基于错误码的代码之间的差异:不差的基于错误码的代码检查错误码。 差的基于错误码的代码从不检查错误码。诚然,很难判断错误是否被正确处理,但至少你可以答出差的代码和不差的代码之间的差异。(这可能不是很好,但至少不差)
另一方面,看出差的基于异常的代码与不差的基于异常的代码之间的区别是非常困难的。
因为,当我编写基于异常的代码时,我没有奢侈地先写差的代码然后再将它们修改成不差的,如果我这样做,我无法找到差的代码,因为它看起来几乎与不差的代码相同。
我的观点并不是异常是差的。我的观点是异常太难了,我不够聪明处理他们。(同样地,好像,书籍的作者也是,即使他们试图教你如何使用异常编程)
(是的,有一些编程模型,如RAII和事物,但很少看到使用这些代码的示例)。
Cleaner, more elegant, and harder to recognize(翻译)的更多相关文章
- Cleaner, more elegant, and harder to recognize (msdn blog)
It appears that some people interpreted the title of one of my rants from many months ago, "Cle ...
- Cleaner, more elegant, and wrong(msdn blog)
Cleaner, more elegant, and wrong Just because you can't see the error path doesn't mean it doesn't e ...
- Cleaner, more elegant, and wrong(翻译)
Cleaner,more elegant,and wrong 整洁,更优雅,但是错的 并不是因为你看不到错误的产生路径就意味着它不存在. 下面是C#编程书中的一个片段,摘自关于异常处理的章节. try ...
- Go 开发关键技术指南 | 敢问路在何方?(内含超全知识大图)
作者 | 杨成立(忘篱) 阿里巴巴高级技术专家 Go 开发关键技术指南文章目录: 为什么你要选择 Go? Go 面向失败编程 带着服务器编程金刚经走进 2020 年 敢问路在何方? Go 开发指南大图 ...
- Go 开发关键技术指南 | Go 面向失败编程 (内含超全知识大图)
作者 | 杨成立(忘篱) 阿里巴巴高级技术专家 关注"阿里巴巴云原生"公众号,回复 Go 即可查看清晰知识大图! 导读:从问题本身出发,不局限于 Go 语言,探讨服务器中常常遇到的 ...
- C# Development 13 Things Every C# Developer Should Know
https://dzone.com/refcardz/csharp C#Development 13 Things Every C# Developer Should Know Written by ...
- 【转载】Gradle学习 第六章:构建脚本基础
转载地址:http://ask.android-studio.org/?/article/11 6.1. Projects and tasks 项目和任务Everything in Gradle si ...
- PEP8中文翻译(转)
原文:https://github.com/zgia/manual PEP 8 -- Style Guide for Python Code PEP Index > PEP 8 -- Style ...
- LLVM 编码规范 - 中文翻译
LLVM 编码规范 导论 语言.库和标准 C++ 标准版本 C++ 标准库 Go 代码准则 机械的代码问题 代码格式化 注释 头文件 类概述 method information 注释格式化 使用Do ...
随机推荐
- thinkphp 使用插件异步上传图片或者文件
使用tp做一些上传的功能,的确挺方便.但是在一些特殊情况下无法单独的使用tp的上传功能, 或者需要做一些比较酷炫的上传效果,这里就需要用到框架了. 我在这里使用的是uploadify上传插件. 首先需 ...
- 小白关于python 对象和内存的关系的一些感悟和疑惑,望大神指教
首先你输入了一个字符串,这个字符串是有大小的,电脑将其放在内存中,自动给其一个起始指针指向这个字符串的首位置,然后,你将这个字符串赋值给一个变量,这个对象又在内存中开辟出一个空间,这个变量会自动连接这 ...
- web更改AD用户密码
web更改AD用户密码 #web更改AD密码 #网站配置 绑定域名ad.test.cn 功能,更改AD用户密码 #参考http://bbs.51cto.com/thread-1379675-1.htm ...
- centos 下安装pptp (vpn) 的方法
废话少说 01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 3 ...
- CORS预检请求详谈
引言 最近在项目中因前后端部署不同地方,前端在请求后端api时发生了跨域请求,我们采用CORS(跨域资源共享)来解决跨域请求,这需要前后端的配合来完成.在这一过程中,后端支持了CORS跨域请求后,前端 ...
- 速微共享链的使用步骤和源码分析(UI设计参考)
一.速微共享链引言 速微共享链Service服务是Android四大组件之一,在Android中有着举足重轻的作用.Service服务是工作的UI线程中,当你的应用需要下载一个文件或者播放音乐等长期处 ...
- Material使用03 MdCardModule模块、MdInputModule模块
需求:先需要增加一个登录模块 1 创建登录模块 ng g m testLogin 1.1 将共享模块导入到登录模块中 import { NgModule } from '@angular/core'; ...
- Navicat for Mysql 暴力破解教程
关于破解Navicat for MySQL的教程有很多 ,但是比较繁琐, 这里推荐一种比较简单的办法~ 网盘地址:链接: https://pan.baidu.com/s/1kVHyShL 密码: ws ...
- 2.python的文件类型、变量数值和字符串练习
1.python的文件类型 .源代码 -python 源代码文件以"py"为扩展名,由python程序解释,不需要编译. 2.字节代码(编译的) -python源码文件经编译后生成 ...
- 理解vuex的状态管理模式架构
理解vuex的状态管理模式架构 一: 什么是vuex?官方解释如下:vuex是一个专为vue.js应用程序开发的状态管理模式.它采用集中式存储管理应用的所有组件的状态,并以相应的规则保证以一种可预测的 ...