`
44424742
  • 浏览: 233106 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
文章分类
社区版块
存档分类
最新评论

浅谈Java语言接口与继承本质区别(转载)

阅读更多
作者:北大青鸟 发表时间:2010-2-24 22:30:46
大多数人认为,接口的意义在于顶替多重继承。众所周知Java没有c 那样多重继承的机制,但是却能够实作多个接口。其实这样做是很牵强的,接口和继承是完全不同的东西,接口没有能力代替多重继承,也没有这个义务。接口的作用,一言以蔽之,就是标志类的类别(typeofclass)。把不同类型的类归于不同的接口,可以更好的管理他们。OO的精髓,我以为,是对对象的抽象,最能体现这一点的就是接口。为什么我们讨论设计模式都只针对具备了抽象能力的语言(比如c 、java、c#等),就是因为设计模式所研究的,实际上就是如何合理的去抽象。(cowboy的名言是“抽象就是抽去像的部分”,看似调侃,实乃至理)。
  设计模式中最基础的是工厂模式(Factory),在我最近的一个很简单的应用中,我想尽量的让我的程序能够在多个数据库间移植,当然,这涉及很多问题,单是如何兼容不同DBMS的SQL就让人头痛。我们不妨先把问题简单化,只考虑如何连接不同的数据库。
  假设我有很多个类,分别是Mysql.java、SQLServer.java、Oracle.java、DB2.java,他们分别连接不同的数据库,统一返回一个Connection对象,并且都有一个close方法,用于关闭连接。只需要针对你的DBMS,选择不同的类,就可以用了,但是我的用户他会使用什么数据库?我不知道,我希望的是尽量少的修改代码,就能满足他的需要。我可以抽象如下接口: 
packageorg.bromon.test;
  publicinterfaceDB
  {
  java.sql.ConnectionopenDB(Stringurl,Stringuser,Stringpassword);
  voidclose();
  }
  这个接口只定义两个方法,没有任何有实际意义的代码,具体的代码由实作这个接口的类来给出,比如Mysql.java:
  Packageorg.bromon.test;
  importjava.sql.*;
  publicclassMysqlimplementsDB
  {
  privateStringurl=”jdbc:mysql:localhost:3306/test”;
  privateStringuser=”root”;
  privateStringpassword=””;
  privateConnectionconn;
  publicConnectionopenDB(url,user,password)
  {
  //连接数据库的代码
  }
  publicvoidclose()
  {
  //关闭数据库
  }
  }
  类似的当然还有Oracle.java等等,接口DB给这些类归了个类,在应用程序中我们这样定义对象:
  org.bromon.test.DBmyDB;
使用myDB来操作数据库,就可以不用管实际上我所使用的是哪个类,这就是所谓的“开-闭”原则。但是问题在于接口是不能实例化的,myDB=newDB(),这样的代码是绝对错误的,我们只能myDB=newMysql()或者myDB=newOracle()。麻烦了,我还是需要指定具体实例化的是哪个类,用了接口跟没用一样。所以我们需要一个工厂: 
packageorg.bromon.test;
  publicclassDBFactory
  {
  publicstaticDBConnectiongetConn()
  {
  Return(newMysql());
  }
  }
  所以实例化的代码变成:myDB=DBFactory.getConn();
  这就是23种模式中最基础的普通工厂(Factory),工厂类负责具体实例化哪个类,而其他的程序逻辑都是针对DB这个接口进行操作,这就是“针对接口编程”。责任都被推卸给工厂类了,当然你也可以继续定义工厂接口,继续把责任上抛,这就演变成抽象工厂(AbstractFactory)。
  整个过程中接口不负责任何具体操作,其他的程序要连接数据库的话,只需要构造一个DB对象就OK,而不管工厂类如何变化。这就是接口的意义----抽象。
继承的概念不用多说,很好理解。为什么要继承呢?因为你想重用代码?这绝对不是理由,继承的意义也在于抽象,而不是代码重用。如果对象A有一个run()方法,对象B也想有这个方法,所以有人就ClassBextendsA。这是不经大脑的做法。如果在B中实例化一个A,调用A的Run()方法,是不是可以达到同样的目的?如下:  
ClassB
  {
  Aa=newA();
  a.run();
  }
  这就是利用类的聚合来重用代码,是委派模式的雏形,是GoF一贯倡导的做法。
  那么继承的意义何在?其实这是历史原因造成的,最开始的OO语言只有继承,没有接口,所以只能以继承来实现抽象,请一定注意,继承的本意在于抽象,而非代码重用(虽然继承也有这个作用),这是很多Java烂书最严重的错误之一,它们所造成的阴影,我至今还没有完全摆脱,坏书害人啊,尤其是入门类的,流毒太大。什么时候应该使用继承?只在抽象类中使用,其他情况下尽量不使用。抽象类也是不能实例化的,它仅仅提供一个模版而已,这就很能说明问题。
  软件开发的万恶之源,一是重复代码而不是重用代码,二是烂用继承,尤以c 程序员为甚。Java中取缔多重继承,目的就是制止烂用继承,实是非常明智的做法,不过很多人都不理解。Java能够更好的体现设计,这是让我入迷的原因之一。
分享到:
评论

相关推荐

    浅谈Java面向接口编程

    浅谈Java面向接口编程 面向接口编程是Java编程中的一种重要思想,它与面向对象编程紧密相关。在本文中,我们将探讨面向接口编程的概念、接口的本质、面向接口编程的思想内涵、面向接口编程的好处等方面。 一、面向...

    浅谈计算机编程语言的发展.pdf

    Eiffel、C++和Java等语言支持数据隐藏、数据抽象、继承和多态等特性,使程序员能够以更符合问题本质的方式来描述和解决问题,极大地提高了软件的可维护性和复用性。 4. **现代编程语言的形式** 现代编程语言主要...

    浅谈java的守护线程与非守护线程

    浅谈java的守护线程与非守护线程 在Java中,有两类线程:UserThread(用户线程)和Daemon Thread(守护线程)。守护线程的作用是为其他线程的运行提供服务,比如说GC线程。它们的本质上来说没有区别,唯一的区别...

    浅谈三分钟学习Java泛型中T、E、K、V、?的含义

    Java 泛型是 Java SE 1.5 中引入的一项新特性,泛型的本质是参数化类型,也就是说所操作的数据类型被指定为一个参数。这种参数类型可以用在类、接口和方法的创建中,分别称为泛型类、泛型接口、泛型方法。 泛型的...

    浅谈JSP serverlet的区别与联系

    JSP(JavaServer Pages)和Servlet是Java Web开发中两个重要的技术,它们都用于构建动态Web页面,但各自有不同的特点和用途。以下是关于JSP和Servlet的区别与联系的知识点详细阐述: 1. JSP的基本概念和特点: - ...

    Java 设计模式

    - **从招式与内功谈起**:这部分内容旨在通过比喻的方式,帮助读者理解设计模式的本质及其在实际开发中的应用价值。招式指的是具体的编码技巧,而内功则是对设计模式背后的原理的理解。 #### 三、面向对象设计原则 ...

    浅谈Servlet开发技术基础

    Servlet本质上是一个Java小程序,需要实现Servlet接口,以便服务器能够识别并调用。在Servlet的开发过程中,主要有以下几个关键点: 1. **Servlet的创建与初始化**: - 当Servlet首次被请求时,Web容器(如Tomcat...

    菜鸟的自我修炼——阿里巴巴一道笔试题浅谈

    2. **面向对象**:Java是一种面向对象的语言,因此对类、对象、继承、封装和多态的理解至关重要。笔试题可能会设计一个简单的类结构,要求你实现特定功能或者分析代码行为。 3. **集合框架**:Java集合框架包括...

    我的编程感悟(中文PDF)(共37M二分卷)分卷二

    6.1 浅谈代码优化 138 6.2 并不仅仅是汇编 139 6.2.1 在算法实现时减少上下文的依赖关系 139 6.2.2 低效的静态变量 140 6.2.3 数据的组织 141 6.2.4 消除除法 142 6.2.5 避免过大的循环 144 6.3 汇编和C/C++的混合...

    我的编程感悟(中文PDF)(共37M二分卷)分卷一

    6.1 浅谈代码优化 138 6.2 并不仅仅是汇编 139 6.2.1 在算法实现时减少上下文的依赖关系 139 6.2.2 低效的静态变量 140 6.2.3 数据的组织 141 6.2.4 消除除法 142 6.2.5 避免过大的循环 144 6.3 汇编和C/C++的混合...

Global site tag (gtag.js) - Google Analytics