`

澄清Java语言接口与继承的本质(转)

阅读更多
大多数人认为,接口的意义在于顶替多重继承。众所周知Java没有c++那样多重继承的机制,但是却能够实作多个接口。其实这样做是很牵强的,接口和继承是完全不同的东西,接口没有能力代替多重继承,也没有这个义务。接口的作用,一言以蔽之,就是标志类的类别(type of class)。把不同类型的类归于不同的接口,可以更好的管理他们。OO的精髓,我以为,是对对象的抽象,最能体现这一点的就是接口。为什么我们讨论设计模式都只针对具备了抽象能力的语言(比如c++、java、c#等),就是因为设计模式所研究的,实际上就是如何合理的去抽象。(cowboy的名言是“抽象就是抽去像的部分”,看似调侃,实乃至理)。

  设计模式中最基础的是工厂模式(Factory),在我最近的一个很简单的应用中,我想尽量的让我的程序能够在多个数据库间移植,当然,这涉及很多问题,单是如何兼容不同DBMS的SQL就让人头痛。我们不妨先把问题简单化,只考虑如何连接不同的数据库。

  假设我有很多个类,分别是Mysql.java、SQLServer.java、Oracle.java、DB2.java,他们分别连接不同的数据库,统一返回一个Connection对象,并且都有一个close方法,用于关闭连接。只需要针对你的DBMS,选择不同的类,就可以用了,但是我的用户他会使用什么数据库?我不知道,我希望的是尽量少的修改代码,就能满足他的需要。我可以抽象如下接口:

package org.bromon.test;<br />

public interface DB<br />
{
  java.sql.Connection openDB(String url,String user,String password);
  void close();
}



  这个接口只定义两个方法,没有任何有实际意义的代码,具体的代码由实作这个接口的类来给出,比如Mysql.java:

Package org.bromon.test
import java.sql.*;

public class Mysql implements DB
{
  private String url=”jdbc:mysql:localhost:3306/test”;
  private String user=”root”;
  private String password=””;
  private Connection conn;
  public Connection openDB(url,user,password)
  {
    //连接数据库的代码

  }
  public void close()
  {
    //关闭数据库
  }
}


  类似的当然还有Oracle.java等等,接口DB给这些类归了个类,在应用程序中我们这样定义对象:

  org.bromon.test.DB myDB;

  使用myDB来操作数据库,就可以不用管实际上我所使用的是哪个类,这就是所谓的“开-闭”原则。但是问题在于接口是不能实例化的,myDB=new DB(),这样的代码是绝对错误的,我们只能myDB=new Mysql()或者myDB=new Oracle()。麻烦了,我还是需要指定具体实例化的是哪个类,用了接口跟没用一样。所以我们需要一个工厂:

package org.bromon.test;
public class DBFactory
{
  public static DB Connection getConn()
  {
    Return(new Mysql());

  }
}



  所以实例化的代码变成:myDB=DBFactory.getConn();

  这就是23种模式中最基础的普通工厂(Factory),工厂类负责具体实例化哪个类,而其他的程序逻辑都是针对DB这个接口进行操作,这就是“针对接口编程”。责任都被推卸给工厂类了,当然你也可以继续定义工厂接口,继续把责任上抛,这就演变成抽象工厂(Abstract Factory)。

  整个过程中接口不负责任何具体操作,其他的程序要连接数据库的话,只需要构造一个DB对象就OK,而不管工厂类如何变化。这就是接口的意义----抽象。

  继承的概念不用多说,很好理解。为什么要继承呢?因为你想重用代码?这绝对不是理由,继承的意义也在于抽象,而不是代码重用。如果对象A有一个run()方法,对象B也想有这个方法,所以有人就Class B extends A。这是不经大脑的做法。如果在B中实例化一个A,调用A的Run()方法,是不是可以达到同样的目的?如下:

Class B
{
  A a=new A();
  a.run();
}

  这就是利用类的聚合来重用代码,是委派模式的雏形,是GoF一贯倡导的做法。

  那么继承的意义何在?其实这是历史原因造成的,最开始的OO语言只有继承,没有接口,所以只能以继承来实现抽象,请一定注意,继承的本意在于抽象,而非代码重用(虽然继承也有这个作用),这是很多Java烂书最严重的错误之一,它们所造成的阴影,我至今还没有完全摆脱,坏书害人啊,尤其是入门类的,流毒太大。什么时候应该使用继承?只在抽象类中使用,其他情况下尽量不使用。抽象类也是不能实例化的,它仅仅提供一个模版而已,这就很能说明问题。

  软件开发的万恶之源,一是重复代码而不是重用代码,二是烂用继承,尤以c++程序员为甚。Java中取缔多重继承,目的就是制止烂用继承,实是非常明智的做法,不过很多人都不理解。Java能够更好的体现设计,这是让我入迷的原因之一。
分享到:
评论

相关推荐

    澄清Java(接口与继承)

    ### 澄清Java中的接口与继承 在Java编程语言中,接口与继承是实现面向对象编程的关键概念。本文将详细解析这两个概念及其在实际开发中的应用,帮助开发者更好地理解和运用它们。 #### 一、接口的理解与应用 #####...

    java六大必须澄清的概念-常考题目

    Java只支持单一继承,但通过接口可以实现多重继承的效果。 4. **多态**: 多态是指同一操作作用于不同的对象,可以有不同的解释,产生不同的执行结果。Java中的多态主要体现在方法的覆盖和接口的实现。通过向上...

    java面向对象--图形继承关系报告[文].pdf

    综上所述,文件内容从标题到描述,再到部分内容,紧密围绕Java面向对象编程的继承特性展开,介绍了类与接口的设计,以及通过继承关系构建图形类库的基本思路。在实际开发中,这样的结构能够提高代码的可维护性和可...

    Java语言程序设计 郑莉 王行言 马素霞 课后习题答案

    《Java语言程序设计》是由郑莉、王行言和马素霞三位专家共同编著的一本深入探讨Java编程的教材。这本书涵盖了从基础到高级的Java编程知识,并且每个章节都配备了丰富的课后习题,旨在帮助读者巩固所学概念并提升实践...

    java sql javad

    ### Java与SQL基础知识点概述 #### 一、Java简介 Java是一种广泛使用的高级编程语言,由Sun Microsystems在1995年首次发布。它被设计为面向对象的语言,并且具有平台独立性,这意味着编写的Java代码可以在任何支持...

    对JAVA语言的十个常见误解

    ### 对JAVA语言的十个常见误解 #### 误解一:JAVA是一种HTML编程技术 很多人误以为JAVA与HTML有着紧密的联系,甚至将JAVA视为一种HTML编程技术。实际上,JAVA与HTML是两种完全不同的技术。HTML主要用于定义网页的...

    JavaJava%BEE软件工程师就业求职手册.doc

    第二章则详细阐述了Java语言基础,包括类和对象、命名规范、继承、常用命令以及编程原则,这些都是Java编程的基础。 第三章进一步探讨了Java语言的细节,如标识符、关键字、注释、原始类型等,帮助读者巩固对基本...

    Java的考试试卷

    16. **单继承与多重继承**:Java中,类的继承遵循单一继承原则,即一个类只能继承一个父类,但可以通过实现多个接口来模拟多重继承。 17. **线程实现方式**:Java线程可以通过继承`Thread`类或者实现`Runnable`接口...

    java 基础及入门

    #### Java与JavaScript的区别 尽管名称相似,Java和JavaScript在语言设计、用途和执行环境上有着根本的不同。Java是一种静态类型的、面向对象的编程语言,通常用于构建大型企业级应用、桌面应用程序和安卓移动应用...

    Java开发注意事项

    在深入探讨Java开发注意事项之前,我们首先应当澄清,给定的部分内容似乎包含了非文本或乱码信息,这可能源于文档转换或编码问题。不过,基于标题“Java开发注意事项”及描述“Java开发注意事项,编程人员可以看看哦...

    java基础知识

    #### 构造器、抽象类与接口的深入探讨 - **构造器(Constructor)**:不能被重写,但可以被重载。构造器主要用于初始化对象的状态,每个类至少有一个构造器,如果没有显式定义,Java会提供一个默认的无参构造器。 ...

    我当老师的180天_Java入门

    这一阶段需要强调Java与C++等其他面向对象语言的区别。 - 第二阶段:使用Java进行图形用户界面(GUI)编程。这部分将引导学生从传统的可视化编程过渡到手动创建界面。 - **额外知识点**:考虑到课程时长充裕,可以...

    《Java 核心技术 卷I 》第12版,涵盖Java17新特性(英文版)

    这一章还讨论了Java applet和互联网的关系,以及Java的历史,同时澄清了一些关于Java的常见误解。 第二章详细阐述了Java编程环境的设置,包括安装Java Development Kit (JDK),下载与设置JDK,以及安装源文件和文档...

    JAVA教程之线程篇

    - **线程的创建与管理**:在Java中,线程可以通过继承`Thread`类或实现`Runnable`接口来创建。通过调用`start()`方法启动线程,而`run()`方法则包含了线程执行的具体逻辑。Java提供了丰富的API,如`Thread`类和`...

    java_disabuse.rar_java disabuse

    "java_disabuse.rar_java disabuse"这个压缩包文件旨在帮助解决这些问题,通过实例来澄清常见的Java学习误区。 文档“java_disabuse.doc”可能包含了以下几个方面的重要知识点: 1. **基础语法**:Java的基础语法...

    java概念技术地图

    6. Java技术图提供了一个全面的视角,通过概念图的形式,展示了整体结构和细节,有助于人们构建和澄清心智模型。 7. 站点***提供更多信息,虽然该链接现在可能已经过时,但它在过去是获取Java相关知识的官方站点。 ...

    什么是Java EE 5

    ### Java EE 5:企业级Java应用的革新与简化 #### 概述 Java EE 5,即Java Platform Enterprise Edition 5,标志着企业级Java应用的一次重要进化。这一版本的发布不仅重命名了前代的J2EE 1.5,而且在功能和架构上...

    臧圩人--JAVA面试题解惑系列合集.pdf

    - **知识点**:澄清Java中参数传递的机制,即所有参数都是按值传递,但对于对象而言,传递的是指向该对象的引用的值,而非对象本身。 **1.6 JAVA面试题解惑系列(六)——字符串(String)杂谈** - **知识点**:...

Global site tag (gtag.js) - Google Analytics