- 浏览: 5164218 次
- 性别:
- 来自: 天津
博客专栏
-
实战 Groovy
浏览量:29320
文章分类
- 全部博客 (639)
- 代码之谜 (6)
- JavaScript quirks (5)
- 程序员 (92)
- Java (93)
- BT编程 (7)
- html/css (64)
- Groovy&Grails (42)
- Android (20)
- C/C++ (5)
- PHP/Perl/Python (46)
- 经典文章 (51)
- CodeIgniter (14)
- JQuery (10)
- 笑话 (4)
- 其他 (32)
- javascript (69)
- 云计算 (0)
- html5 (7)
- 面试 (8)
- google (3)
- nosql (2)
- nodejs (11)
- go (5)
- erlang (1)
- 小常识 (3)
- 冷知识 (5)
- database (4)
- web (12)
- 架构 (12)
- Exception (0)
最新评论
-
jqw1992:
https://www.chromefor.com/libra ...
[福利] 开发者必备的 Chrome 插件——ChromeSnifferPlus -
litjerk:
初步算了一下,目前最最精简的Win98版是5M,他5个小时多敲 ...
让人目瞪口呆的三位世界级电脑大师 -
379855529:
。。似乎重点没说NIO啊,前面基础只是铺垫的很好的,可是我要的 ...
Java NIO与IO的详细区别(通俗篇) -
springmvc_springjpa:
spring mvc demo教程源代码下载,地址:http: ...
一步步开发 Spring MVC 应用 -
匡建武:
Good
四个程序员的一天
我们有很多 Coding Style 或 代码规范。 但这一条可能会经常被我们所遗忘,就是我们 经常会在函数的参数里使用bool参数,这会大大地降低代码的可读性。 不信?我们先来看看下面的代码。
当你读到下面的代码,你会觉得这个代码是什么意思?
widget->repaint(false);
是不要 repaint 吗?还是别的什么意思?
看了文档后,我们才知道这个参数是 immediate,也就是说,false 代表不立即重画,true 代码立即重画。
Windows API 中也有这样一个函数:InvalidateRect,当你看到下面的代码,你会觉得是什么意思?
InvalidateRect(hwnd, lpRect, false);
我们先不说 InvalidateRect 这个函数名取得有多糟糕,我们先说一下那个 false 参数? invalidate 意为 “让XXX无效”,false 是什么意思? 双重否定? 是肯定的意思?
如果你看到这样的代码,你会相当的费解的。 于是,你要去看一下文档,或是 InvalidateRect 的函数定义, 你会看到那个参数是 BOOL bErase,意思是:“是否要重画背景”。
这样的事情有很多,再看下面的代码,想把 str 中的 ”%USER%” 替换成真实的用户名:
str.replace("%USER%", user, false); // Qt 3
TNND,那个 false 是什么意思?不替换吗?还是别的什么意思?
看了文档才知道,false 代表: “大小写不敏感的替换”。
其实,如果你使用枚举变量/常量,而不是 bool 变量,你会让你的代码更易读,如:
widget->repaint(PAINT::immediate);
widget->repaint(PAINT::deffer);
InvalidateRect(hwnd, lpRect, !RepantBackground);
str.replace("%USER%", user, Qt::CaseInsensitive); // Qt 4
如果对这个事不以为然的话,我们再来看一些别的示例,你不妨猜猜看看下面的代码:
component.setCentered(true, false);
这什么玩意儿啊?
看了文档你才知道,这原来是 setCentered(centered, autoUpdate);
new Textbox(300, 100, false, true);
这又是什么啊?
看了文档才知道,这是创建一个文本框,第三个参数是:“是否要滚动条”,第四个是:“是否要自动换行”。TNND!
这种情况还不算最差,看看下面的双重否定。
component.setDisabled(false);
filter.setCaseInsensitive(false)
再来一个,如果你读到下面的代码,相信你会和我一样,要么石化了,要么凌乱了。
event.initKeyEvent("keypress", true, true, null, null,
false, false, false, false, 9, 0);
看完这篇文章,我希望你再也 不要把bool为作为函数参数了。除非两个原因:
- 你 100% 确认不会带来阅读上的问题,比如 Java 的
setVisible (bool)
. - 你 100% 确认你想去 写出迷一样的代码。
如果你想设计一个好的 API,强烈推荐你读一下 Nokia 的 Qt 的《API Design Principles》,本文就是其中的 “Boolean Trap”。
原文链接:http://coolshell.cn/articles/5444.html
评论
另外,像这样的重点难道没注释!这才是最坑爹的地方!楼主重点没搞对的说,所以你上面的一大堆几乎算是废话,不过也挺喜欢这样的废话,很有趣。
以上尽是愚人短见
。。。
不是说用boolean做参数就一定会产生控制耦合(例如setter),但控制耦合往往会产生boolean参数。因此看到boolean参数就最好特别留意一下,看看背后是不是跟着一堆控制耦合。
5.3.2 不可取的内聚性
其余类型的内聚性,一般来说都是不可取的。其后果往往是产生一些组织混乱而又难以调
试和改进的代码。如果一个子程序具有不良的内聚性,那最好重新创建一个较好的子程序,而不要去试图修补它。知道应该避免什么是非常重要的,以下就是一些不可取的内聚性:
...
逻辑内聚性。当一个子程序中同时含有几个操作,而其中一个操作又被传进来的控制标志
所选择时,就产生了逻辑内聚性。之所以称之为逻辑内聚性,是因为这些操作仅仅是因为控制流,或者说“逻辑”的原因才联系到一起的,它们都被包括在一个很大的 if 或者 case 语句中,它们之间并没有任何其它逻辑上的联系。
举例来说,一个叫作 InputAll()的子程序,程序的输入内容可能是用户名字、雇员时间卡信息或者库存数据,至于到底是其中的哪一个,则由传入子程序的控制标志决定。其余类似的子程序还有 ComputeAll(),EditAll(),PrintAll()等等。这类子程序的主要问题是一定要通过传入一个控制标志来决定子程序处理的内容。解决的办法是编写三个不同的子程序,每个子程序只进行其中一个操作。如果这三个子程序中含有公共代码段,那么还应把这段代码放入一个较低层次的子程序中,以供三个子程序调用。并且,把这三个子程序放入一个模块中。
但是,如果一个逻辑内聚性的子程序代码都是一系列 if 和 case 语句,并且调用其它子程序,那么这是允许的。在这种情况下,如果程序的唯一功能是调度命令,而它本身并不进行任何处理,那么这可以说是一个不错的设计。对这种子程序的专业叫法是“事物处理中心”,事物处理中心往往被用作基础环境下的事件处理,比如,Apple Macintosh 和 Microsoft Windows。
......
控制耦合。如果一个子程序通过传入另一个子程序的数据通知它该作什么,那么这两个子
程序就是控制耦合的。控制耦合是令人不快的,因为它往往与逻辑内聚性联在一起,并且,通常都要求调用程序了解被调子程序的内容与结构。
摘自《完全代码》(《Code Complete》)第五章 《高质量子程序的特点》
public String getTrueNum(){ getNum(true); } public String getFalseNum(){ getNum(false); } private String getNum(Boolean)
作为方法的定义处,使用bool类型的形参并没有问题。
比如 public void repaint(boolean immediate)这种。反正自动代码补充时都会生成类似于widget.repaint(immediate)这样的代码人们很容易就能明白形参要表示什么意思。当然如果看它的文档(JavaDoc等)就更不会有问题了。
作为方法的使用处,如果怕使用widget.repaint(false)这样的实参以后,代码别人看不懂,只需要写成这样就可以了
boolean immediate=false;
widget.repaint(immediate);
boolean disabled=false;
component.setDisabled(disabled);
widget.repaint(immediate=false)
作为方法的定义处,使用bool类型的形参并没有问题。
比如 public void repaint(boolean immediate)这种。反正自动代码补充时都会生成类似于widget.repaint(immediate)这样的代码人们很容易就能明白形参要表示什么意思。当然如果看它的文档(JavaDoc等)就更不会有问题了。
作为方法的使用处,如果怕使用widget.repaint(false)这样的实参以后,代码别人看不懂,只需要写成这样就可以了
boolean immediate=false;
widget.repaint(immediate);
boolean disabled=false;
component.setDisabled(disabled);
加注释也有潜在问题的。一般情况下,如果这个方法的内部实现就是一个最外层的if套着两段完全不同的执行逻辑,那么在一开始你就会把它拆分成两个方法。比如说markForRepaint和repaintImmediately。人们之所以传入一个boolean作为控制标志,往往是因为方法的内部实现有大量公共部分,里面嵌着若干依赖于这个标志的分支。
这种方法最好在一开始的时候就想办法拆开,因为你在第一次开发时通常有比较充足的时间来重构,思路也比较连贯。否则在后续维护时,某人可能想复用这个方法,不过要做点修改。如果你本来就拆开了,他会按照你的思路来改。如果你本来就用boolean参数来控制分支,他可能就匆匆忙忙地想在其中加入一个新的分支,这时他会再加入一个新的boolean参数。如果这两个boolean参数完全独立那暂时还好,最怕是前一个为真时后一个完全无效,只有前一个为假时才有效。或者更痛苦的是前一个为真或假时,后一个的真假分别有细微的差异。然后再过段时间,又有人加进去一个boolean参数……
换句话说,你提出了一个问题,但同时我希望你也能够给出一个解决这个问题的提案,这个提案不一定需要正确,但是多少给我们一个思考的方向。
顺便说一句,看见问题的人很多,但是解决问题的人很少。所以很多问题存在,大家也都知道,却都没有去解决,很可能跟着有关。
如果传入的boolean被当成普通数据来处理(例如直接保存到数据库中),其实是没有问题的。问题是绝大多数情况下,传入的boolean被用在if里作为分支条件,这就构成了在《Code Complete》里所谓的“控制耦合”,函数A通过传入某种标志来控制函数B的执行逻辑。这种耦合的问题是,除非你了解函数B的实现细节(或者有详细的文档),否则很难通过命名去表达清楚传入的控制标志会造成什么后果。而且随着B的执行逻辑不断修改,这个标志的实际效果往往会跟它的命名含义脱节。
解决的方法也有不少,只不过总不如写一个boolean的参数方便,所以很多人事到临头就会偷懒。包括:
1. 利用多态把分支结构转移到继承结构中。这需要有一定的架构能力。
2. 把boolean参数的true和false两种情况拆分成两个方法。(问题是如果公共部分较多,分支又不止一处,传入boolean参数就实在是太方便了。如果语言支持lambda特性重构起来就比较方便,否则就需要用模板模式或者策略模式来重构,相对比较麻烦)
3. 用enum类型(如果语言不支持,可以用命名常量代替,但效果不如enum)来做控制标志比用boolean要好。这种方案最常用,不过没有解决随着时间推移,标志的字面含义容易与实际行为脱节的问题。
陈浩哥的。
换句话说,你提出了一个问题,但同时我希望你也能够给出一个解决这个问题的提案,这个提案不一定需要正确,但是多少给我们一个思考的方向。
顺便说一句,看见问题的人很多,但是解决问题的人很少。所以很多问题存在,大家也都知道,却都没有去解决,很可能跟着有关。
发表评论
-
Java NIO与IO的详细区别(通俗篇)
2014-01-02 10:51 69420内核空间、用户空间 ... -
Java EE 单元测试
2013-09-12 10:56 4777觉得测试 Java EE 应用程序太困难、不方便或者太复杂? ... -
JUnit 测试建议
2013-07-22 09:04 3480以下是对JUnit实践的一 ... -
老生常谈:面试算法有必要吗?
2013-04-01 09:55 32753声明:虽然发表在愚人节,但是文章内容很严肃。 前几天在知 ... -
一步步开发 Spring MVC 应用
2013-02-23 10:53 10350Spring MVC 框架 Spring 框 ... -
「译」Java集合框架系列教程四:Set接口
2013-01-18 13:20 2529原文:The Set Interface 译文:Java集 ... -
「译」Java集合框架系列教程三:Collection接口
2013-01-18 13:16 2369原文:The Collection Interface 译 ... -
「译」Java集合框架系列教程二:集合接口
2013-01-18 13:13 2011原文:http://docs.oracle.c ... -
Java集合框架系列教程一:集合框架简介
2013-01-18 12:47 3028原文:http://docs.oracle.com/jav ... -
Java泛型-类型擦除
2012-12-05 15:48 14615一、概述 Java泛型在使用过程有诸多的 ... -
网站建设中关于eclipse启动参数的优化
2012-08-23 09:23 2501在网站建设中,经常用 ... -
Eclipse程序员要掌握的常用快捷键
2012-08-22 09:33 10060判断一个人的编程水平,就看他用键盘多,还是鼠标多。用键 ... -
java中equals和==的区别
2012-08-15 16:31 2029值类型是存储在内存中的堆栈(简称栈),而引用类型的变量在 ... -
名词王国里的死刑(翻译) - A Story of Hello World
2012-07-23 10:43 11811翻译自Steve Yegge的大 ... -
好代码是廉价的代码
2012-07-19 08:03 3295长久以来我一直主张:好代码是廉价的代码。 当我跟做开发 ... -
只要一个返回语句
2012-05-18 13:18 1820别再这样写了: publ ... -
java编程的78条黄金法则
2012-05-16 12:57 2358创建和销毁对象 1、考虑用静态工厂方法(返回类的实例的 ... -
for 循环为何可恨?
2012-05-15 12:44 1619Java的闭包(Closure)特征最近成为了一个热门话 ... -
Java 8 新功能
2012-05-02 08:48 1902我们已经急不可待想知道Java 8可用的新功能,其中大部 ... -
java面试题及答案(二)
2012-04-25 13:01 247216、同步和异步有何异 ...
相关推荐
当函数参数为bool时,程序员需要通过查阅文档或源码才能准确理解其含义,这无疑增加了理解和维护代码的难度。 首先,让我们看一个例子,`widget->repaint(false)`。如果不了解`repaint`函数的具体实现,`false`在...
在Python编程语言中,`bool`函数是一种内置的布尔类型转换函数,用于将任何类型的数据转化为对应的布尔值,即`True`或`False`。布尔值在逻辑表达式和条件语句中扮演着核心角色。下面我们将详细探讨`bool`函数在不同...
C#调用DLL中非托管C++函数参数类型对照 在C#编程中,经常需要调用C++中的DLL类库,这就需要了解C++中的函数参数类型在C#中的对应关系。以下是基本数据类型的对照: * 一维数组:C#参数在基本类型前加ref或out,out...
python里面的bool函数应用
C# 调用 C++ 编写的 DLL 函数各种参数传递问题 在 C# 调用 C++ 编写的 DLL 函数时,参数传递是一个非常重要的部分。这篇文章将详细介绍 C# 调用 C++ 编写的 DLL 函数各种参数传递问题,包括不返回值的参数、带...
在Python中,bool类型的取值范围非常有限,严格来说,只有True和False这两种值,但是它可以从不同数据类型的值中转化而来,下面将详细介绍bool函数取值方法的细节。 1. 数字的bool取值方法: 在Python中,当bool...
本文将深入探讨Dart中的函数,包括它们的定义、可选参数、默认参数、命名参数、箭头函数、匿名函数以及闭包等关键概念。 首先,我们来了解**函数的定义**。在Dart中,你可以使用`function`关键字或者通过指定函数体...
1. void AcknowledgeMessage(DWORD MsgNr)函数:确认消息系统中带编号的消息,该编号被传递为参数。该函数可以用来确认选择的报警记录消息。 2. BOOL AXC_OnBtnArcLong(char* lpszPictureName, char* pszMsgWin)...
本手册涵盖了WinCC标准函数的基本概念、使用方法、参数设置、实践示例等多方面的内容,为用户提供了一份详细的参考资料。 Alarm函数是WinCC标准函数中的一个重要组成部分,用于控制WinCC报警控件的行为。Alarm函数...
这将把byData的第3位(按右对齐,最右边的位是第0位)转换为布尔值赋给boolValue。 在SCL编程中,这些转换函数非常实用,尤其是在处理大量的开关量输入/输出或者配置复杂的位逻辑时。为了实现这些转换,SCL提供了...
函数内部修改这些参数的值,从而达到“返回”多个值的效果: ```cpp void getResults(int& a, QString& b, bool& c) { a = ...; b = ...; c = ...; } ``` 在调用这个函数时,提供已初始化的变量供函数修改。...
函数重载是C++中一项非常重要的特性,它允许我们定义具有相同名称但参数列表不同的多个函数。这一特性极大地提高了代码的复用性和可读性。本文将详细介绍函数重载的概念、规则及其在实际开发中的应用。 #### 二、...
重载函数是C++语言中的一个关键特性,它允许在同一个作用域内定义多个同名但参数列表不同的函数,这是C++灵活性与强大性的体现之一。重载函数(Function Overloading)解决了函数名称冲突的问题,同时也提高了代码的...
该函数的原型为:BOOL Create( LPCTSTR lpszCaption, DWORD dwStyle, const RECT& rect, CWnd* pParentWnd, UINT nID );其中,lpszCaption 是按钮上的文字,dwStyle 是按钮的样式,rect 是按钮的矩形区域,...
文件操作基础API函数 文件操作是 Windows 应用程序经常要涉及的内容,利用 Windows 的 API 函数,可以很好地实现文件操作的功能。在 Windows 操作系统中,文件操作可以分为文件的创建、打开、读取、写入、关闭等几...
例如,在Windows API函数中,某些函数返回的`BOOL`类型值可能会采用-1来表示错误情况。 ```cpp #define TRUE 1 #define FALSE 0 #define ERROR -1 // 某些Windows API函数返回值 BOOL result = GetMessage(....
而`tf`(Test Functions)是一个轻量级的框架,专为Go语言设计,用于实现函数化的参数化测试。本文将深入探讨`tf`框架及其在Go开发中的应用。 ### 1. 函数化测试的概念 函数化测试,也称为参数化测试,是一种测试...
在本文中,我们将探讨如何使用API函数来实现自动安装“王码五笔型输入法”。首先,我们关注到一个名为`ImmIsIME`的API函数,它用于判断指定的句柄是否为IME(Input Method Editor,输入法编辑器)。这个函数在...