从Rob Pike 的 Google+上的一个推看到了一篇叫《Understanding Object Oriented Programming》的文章,我先把这篇文章简述一下,然后再说说老牌黑客Rob Pike的评论。

先看这篇教程是怎么来讲述OOP的。它先给了下面这个问题,这个问题需要输出一段关于操作系统的文字:假设Unix很不错,Windows很差。

这个把下面这段代码描述成是Hacker Solution。(这帮人觉得下面这叫黑客?我估计这帮人真是没看过C语言的代码)

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

public class PrintOS

{

    public static void main(final String[] args)

    {

        String osName = System.getProperty("os.name") ;

        if (osName.equals("SunOS") || osName.equals("Linux"))

        {

            System.out.println("This is a UNIX box and therefore good.") ;

        }

        else if (osName.equals("Windows NT") || osName.equals("Windows 95"))

        {

            System.out.println("This is a Windows box and therefore bad.") ;

        }

        else

        {

            System.out.println("This is not a box.") ;

        }

    }

}

然后开始用面向对象的编程方式一步一步地进化这个代码。

先是以过程化的思路来重构之。

过程化的方案

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

public class PrintOS

{

    private static String unixBox()

    {

        return "This is a UNIX box and therefore good." ;

    }

    private static String windowsBox()

    {

        return "This is a Windows box and therefore bad." ;

    }

    private static String defaultBox()

    {

        return "This is not a box." ;

    }

    private static String getTheString(final String osName)

    {

        if (osName.equals("SunOS") || osName.equals("Linux"))

        {

            return unixBox() ;

        }

        else if (osName.equals("Windows NT") ||osName.equals("Windows 95"))

        {

            return windowsBox() ;

        }

        else

        {

            return defaultBox() ;

        }

    }

    public static void main(final String[] args)

    {

        System.out.println(getTheString(System.getProperty("os.name"))) ;

    }

}

然后是一个幼稚的面向对象的思路。

幼稚的面向对象编程

PrintOS.java

1

2

3

4

5

6

7

public class PrintOS

{

    public static void main(final String[] args)

    {

        System.out.println(OSDiscriminator.getBoxSpecifier().getStatement()) ;

    }

}

OSDiscriminator.java

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

public class OSDiscriminator // Factory Pattern

{

    private static BoxSpecifier theBoxSpecifier = null ;

    public static BoxSpecifier getBoxSpecifier()

    {

        if (theBoxSpecifier == null)

        {

            String osName = System.getProperty("os.name") ;

            if (osName.equals("SunOS") || osName.equals("Linux"))

            {

                theBoxSpecifier = new UNIXBox() ;

            }

            else if (osName.equals("Windows NT") || osName.equals("Windows 95"))

            {

                theBoxSpecifier = new WindowsBox() ;

            }

            else

            {

                theBoxSpecifier = new DefaultBox () ;

            }

        }

        return theBoxSpecifier ;

    }

}

BoxSpecifier.java

1

2

3

4

public interface BoxSpecifier

{

    String getStatement() ;

}

DefaultBox.java

1

2

3

4

5

6

7

public class DefaultBox implements BoxSpecifier

{

    public String getStatement()

    {

        return "This is not a box." ;

    }

}

UNIXBox.java

1

2

3

4

5

6

7

public class UNIXBox implements BoxSpecifier

{

    public String getStatement()

    {

        return "This is a UNIX box and therefore good." ;

    }

}

WindowsBox.java

1

2

3

4

5

6

7

public class WindowsBox implements BoxSpecifier

{

    public String getStatement()

    {

        return "This is a Windows box and therefore bad." ;

    }

}

他们觉得上面这段代码没有消除if语句,他们说这叫代码的“logic bottleneck”(逻辑瓶颈),因为如果你要增加一个操作系统的判断的话,你不但要加个类,还要改那段if-else的语句。

所以,他们整出一个叫Sophisticated的面向对象的解决方案。

OO大师的方案

注意其中的Design Pattern

PrintOS.java

1

2

3

4

5

6

7

public class PrintOS

{

    public static void main(final String[] args)

    {

        System.out.println(OSDiscriminator.getBoxSpecifier().getStatement()) ;

    }

}

OSDiscriminator.java

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

public class OSDiscriminator // Factory Pattern

{

    private static java.util.HashMap storage = new java.util.HashMap() ;

    public static BoxSpecifier getBoxSpecifier()

    {

        BoxSpecifier value = (BoxSpecifier)storage.get(System.getProperty("os.name")) ;

        if (value == null)

            return DefaultBox.value ;

        return value ;

    }

    public static void register(final String key, final BoxSpecifier value)

    {

        storage.put(key, value) ; // Should guard against null keys, actually.

    }

    static

    {

        WindowsBox.register() ;

        UNIXBox.register() ;

        MacBox.register() ;

    }

}

BoxSpecifier.java

1

2

3

4

public interface BoxSpecifier

{

    String getStatement() ;

}

DefaultBox.java

1

2

3

4

5

6

7

8

9

public class DefaultBox implements BoxSpecifier // Singleton Pattern

{

    public static final DefaultBox value = new DefaultBox () ;

    private DefaultBox() { }

    public String getStatement()

    {

        return "This is not a box." ;

    }

}

UNIXBox.java

1

2

3

4

5

6

7

8

9

10

11

12

13

14

public class UNIXBox implements BoxSpecifier // Singleton Pattern

{

    public static final UNIXBox value = new UNIXBox() ;

    private UNIXBox() { }

    public  String getStatement()

    {

        return "This is a UNIX box and therefore good." ;

    }

    public static final void register()

    {

        OSDiscriminator.register("SunOS", value) ;

        OSDiscriminator.register("Linux", value) ;

    }

}

WindowsBox.java

1

2

3

4

5

6

7

8

9

10

11

12

13

14

public class WindowsBox implements BoxSpecifier  // Singleton Pattern

{

    public  static final WindowsBox value = new WindowsBox() ;

    private WindowsBox() { }

    public String getStatement()

    {

        return "This is a Windows box and therefore bad." ;

    }

    public static final void register()

    {

        OSDiscriminator.register("Windows NT", value) ;

        OSDiscriminator.register("Windows 95", value) ;

    }

}

MacBox.java

1

2

3

4

5

6

7

8

9

10

11

12

13

public class MacBox implements BoxSpecifier // Singleton Pattern

{

    public static final MacBox value = new MacBox() ;

    private MacBox() { }

    public  String getStatement()

    {

        return "This is a Macintosh box and therefore far superior." ;

    }

    public static final void register()

    {

        OSDiscriminator.register("Mac OS", value) ;

    }

}

作者还非常的意地说,他加了一个“Mac OS”的东西。老实说,当我看到最后这段OO大师搞出来的代码,我快要吐了。我瞬间想到了两件事:一个是以前酷壳上的《面向对象是个骗局》和 《各种流行的编程方式》中说的“设计模式驱动编程”,另一个我想到了那些被敏捷洗过脑的程序员和咨询师,也是这种德行。

于是我去看了一下第一作者Joseph Bergin的主页,这个Ph.D是果然刚刚完成了一本关于敏捷和模式的书。

Rob Pike的评论

(Rob Pike是当年在Bell lab里和Ken一起搞Unix的主儿,后来和Ken开发了UTF-8,现在还和Ken一起搞Go语言。注:不要以为Ken和Dennis是基友,其实他们才是真正的老基友!)

Rob Pike在他的Google+的这贴里评论到这篇文章——

他并不确认这篇文章是不是搞笑?但是他觉得这些个写这篇文章是很认真的。他说他要评论这篇文章是因为他们是一名Hacker,至少这个词出现在这篇文章的术语中。

他说,这个程序根本就不需要什么Object,只需要一张小小的配置表格,里面配置了对应的操作系统和你想输出的文本。这不就完了。这么简单的设计,非常容易地扩展,他们那个所谓的Hack Solution完全就是笨拙的代码。后面那些所谓的代码进化相当疯狂和愚蠢的,这个完全误导了对编程的认知。

然后,他还说,他觉得这些OO的狂热份子非常害怕数据,他们喜欢用多层的类的关系来完成一个本来只需要检索三行数据表的工作。他说他曾经听说有人在他的工作种用各种OO的东西来替换While循环。(我听说中国Thoughtworks那帮搞敏捷的人的确喜欢用Object来替换所有的if-else语句,他们甚至还喜欢把函数的行数限制在10行以内)

他还给了一个链接http://prog21.dadgum.com/156.html,你可以读一读。最后他说,OOP的本质就是——对数据和与之关联的行为进行编程。便就算是这样也不完全对,因为:

Sometimes data is just data and functions are just functions.

我的理解

我觉得,这篇文章的例子举得太差了,差得感觉就像是OO的高级黑。面向对象编程注重的是:1)数据和其行为的打包封装,2)程序的接口和实现的解耦。你那怕,举一个多个开关和多个电器的例子,不然就像STL中,一个排序算法对多个不同容器的例子,都比这个例子要好得多得多。老实说,Java SDK里太多这样的东西了。

我以前给一些公司讲一些设计模式的培训课,我一再提到,那23个经典的设计模式和OO半毛钱关系没有,只不过人家用OO来实现罢了。设计模式就三个准则:1)中意于组合而不是继承,2)依赖于接口而不是实现,3)高内聚,低耦合。你看,这完全就是Unix的设计准则

相关阅读:什么是工程师文化?各位工程师是为什么活的?作为一个IT或互联网公司为什么要工程师文化?

相关阅读: 对程序员有用:2017最新能上Google的hosts文件下载及总结网友遇到的各种hosts问题解决方法及配置详解

相关阅读:你一直使用的能FQ的hosts文件原理是什么?怎么配置?|我来科普下

出处 酷 壳 – CoolShell

作者:陈浩

相关BLOG:SUN’S BLOG - 专注互联网知识,分享互联网精神!去看看:www.whosmall.com

从一些简单代码实例彻底理解面向对象编程思想|OOP本质是什么?的更多相关文章

  1. 面向对象编程思想(OOP)

    本文我将从面向对象编程思想是如何解决软件开发中各种疑难问题的角度,来讲述我们面向对象编程思想的理解,梳理面向对象四大基本特性.七大设计原则和23种设计模式之间的关系. 软件开发中疑难问题: 软件复杂庞 ...

  2. Redis:安装、配置、操作和简单代码实例(C语言Client端)

    Redis:安装.配置.操作和简单代码实例(C语言Client端) - hj19870806的专栏 - 博客频道 - CSDN.NET Redis:安装.配置.操作和简单代码实例(C语言Client端 ...

  3. Java 雇员管理小练习(理解面向对象编程)

    在学习集合框架的时候,初学者很容易练习到学生管理系统.雇员管理体统等练习题.在学习集合框架之前,基本上Java基本语法都学完了,集合框架也从侧面的检验对前面学习的理解.下面用一个曾经做过的练习题,回顾 ...

  4. 什么是面向对象编程(OOP)?

    Java 程序员第一个要了解的基础概念就是:什么是面向对象编程(OOP)? 玩过 DOTA2 (一款推塔杀人的游戏)吗?里面有个齐天大圣的角色,欧洲战队玩的很溜,国内战队却不怎么会玩,自家人不会玩自家 ...

  5. 面向对象编程思想(OOP)

    本文我将从面向对象编程思想是如何解决软件开发中各种疑难问题的角度,来讲述我们面向对象编程思想的理解,梳理面向对象四大基本特性.七大设计原则和23种设计模式之间的关系. 软件开发中疑难问题: 软件复杂庞 ...

  6. 泛型编程、STL的概念、STL模板思想及其六大组件的关系,以及泛型编程(GP)、STL、面向对象编程(OOP)、C++之间的关系

    2013-08-11 10:46:39 介绍STL模板的书,有两本比较经典: 一本是<Generic Programming and the STL>,中文翻译为<泛型编程与STL& ...

  7. IT第二十天 - 面向对象编程思想、抽象类、异常处理、程序操作日志记录、本周总结 ★★★

    IT第二十天 上午 面向对象编程思想 1.组装电脑的设计: (1)电脑的组成:显示器+机箱 (2)机箱的组成:电源+主板+硬盘 (3)主板所包含的部件:cpu+内存+PCI接口+usb接口 (4)PC ...

  8. 面向对象编程思想(前传)--你必须知道的javascript

    在写面向对象编程思想-设计模式中的js部分的时候发现很多基础知识不了解的话,是很难真正理解和读懂js面向对象的代码.为此,在这里先快速补上.然后继续我们的面向对象编程思想-设计模式. 什么是鸭子类型 ...

  9. 面向对象编程思想(前传)--你必须知道的javascript(转载)

    原文地址:http://www.cnblogs.com/zhaopei/p/6623460.html阅读目录   什么是鸭子类型 javascript的面向对象 封装 继承 多态 原型 this指向 ...

随机推荐

  1. Java基础知识强化76:正则表达式之替换功能

    1. 替换功能: String类的replaceAll方法,如下: public String replaceAll(String regex, String replacement): 使用给定的r ...

  2. NYOJ528 找球号(三)位运算

    找球号(三) 时间限制:2000 ms  |  内存限制:3000 KB 难度:2   描述 xiaod现在正在某个球场负责网球的管理工作.为了方便管理,他把每个球都编了号,且每个编号的球的总个数都是 ...

  3. 线段树---HDU2795Billboard

    这道题跟第二个题差不多,求单点的最大值. 题目大意:有个高和宽分别为h, w的广告牌, 这个广告牌分成高为 1 的长条, 每条分别能贴长度为wi长度的广告, 输入的n为广告的条数,广告优先贴在最上边和 ...

  4. ICSharpCode.SharpZipLib实现压缩解压缩

    最近,在项目中经常需要处理压缩和解压缩文件的操作.经过查找,发现了ICSharpCode.SharpZipLib.dll ,这是一个完全由c#编写的Zip, GZip.Tar . BZip2 类库,可 ...

  5. CSS中的盒子模型详解

    很多人对盒子模型搞晕头了,下面通过一个简单的代码来分析盒子模型的结构! 为了方便方便观看!在第一个div中画了一个表格,并将其尺寸设置成与div内容大小一样!且设置body的margin和paddin ...

  6. PHP Calendar 函数

    PHP 5 Calendar 函数 函数 描述 cal_days_in_month() 针对指定的年份和历法,返回一个月中的天数. cal_from_jd() 把儒略日计数转换为指定历法的日期. ca ...

  7. 前端/html5效果收藏

    H5应用 9款漂亮的H5效果 8款漂亮的H5效果 36漂亮的button效果 颜色RGB表 省市二级联动

  8. redux-simple 简化版的redux

    作为react的粉丝,当然要吐槽一下react组件通信问题.react的单向数据流是组件通信的一大阻碍,只允许父组件向子组件传值,子组件向父组件传值只能通过父组件向子组件传递回调函数实现.如果在深层次 ...

  9. JQuery 操作DOM

    DOM(Document Object Model—文档对象模型):一种与浏览器, 平台, 语言无关的接口, 使用该接口可以轻松地访问页面中所有的标准组件DOM 操作的分类:    DOM Core: ...

  10. 武汉科技大学ACM:1005: Soapbear and Honey

    Problem Description Soapbear is the mascot of WHUACM team. Like other bears, Soapbear loves honey ve ...