﻿<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:dc="http://purl.org/dc/elements/1.1/" xmlns:trackback="http://madskills.com/public/xml/rss/module/trackback/" xmlns:wfw="http://wellformedweb.org/CommentAPI/" xmlns:slash="http://purl.org/rss/1.0/modules/slash/"><channel><title>BlogJava-tfc2005-随笔分类-java基础</title><link>http://www.blogjava.net/tfc2005/category/1730.html</link><description /><language>zh-cn</language><lastBuildDate>Wed, 28 Feb 2007 03:40:07 GMT</lastBuildDate><pubDate>Wed, 28 Feb 2007 03:40:07 GMT</pubDate><ttl>60</ttl><item><title>澄清Java语言接口与继承的本质</title><link>http://www.blogjava.net/tfc2005/archive/2005/06/13/6028.html</link><dc:creator>佟福春</dc:creator><author>佟福春</author><pubDate>Mon, 13 Jun 2005 02:32:00 GMT</pubDate><guid>http://www.blogjava.net/tfc2005/archive/2005/06/13/6028.html</guid><wfw:comment>http://www.blogjava.net/tfc2005/comments/6028.html</wfw:comment><comments>http://www.blogjava.net/tfc2005/archive/2005/06/13/6028.html#Feedback</comments><slash:comments>0</slash:comments><wfw:commentRss>http://www.blogjava.net/tfc2005/comments/commentRss/6028.html</wfw:commentRss><trackback:ping>http://www.blogjava.net/tfc2005/services/trackbacks/6028.html</trackback:ping><description><![CDATA[<FONT face=宋体>大多数人认为，接口的意义在于顶替多重继承。众所周知Java没有c++那样多重继承的机制，但是却能够实作多个接口。其实这样做是很牵强的，接口和继承是完全不同的东西，接口没有能力代替多重继承，也没有这个义务。接口的作用，一言以蔽之，就是标志类的类别（type&nbsp;of&nbsp;class）。把不同类型的类归于不同的接口，可以更好的管理他们。OO的精髓，我以为，是对对象的抽象，最能体现这一点的就是接口。为什么我们讨论设计模式都只针对具备了抽象能力的语言（比如c++、java、c#等），就是因为设计模式所研究的，实际上就是如何合理的去抽象。（cowboy的名言是“抽象就是抽去像的部分”，看似调侃，实乃至理）。<BR><BR>　　设计模式中最基础的是工厂模式（Factory），在我最近的一个很简单的应用中，我想尽量的让我的程序能够在多个数据库间移植，当然，这涉及很多问题，单是如何兼容不同DBMS的SQL就让人头痛。我们不妨先把问题简单化，只考虑如何连接不同的数据库。<BR><BR>　　假设我有很多个类，分别是Mysql.java、SQLServer.java、Oracle.java、DB2.java，他们分别连接不同的数据库，统一返回一个Connection对象，并且都有一个close方法，用于关闭连接。只需要针对你的DBMS，选择不同的类，就可以用了，但是我的用户他会使用什么数据库？我不知道，我希望的是尽量少的修改代码，就能满足他的需要。我可以抽象如下接口：<BR><BR>package&nbsp;org.bromon.test;<BR>public&nbsp;interface&nbsp;DB<BR>{<BR>　　java.sql.Connection&nbsp;openDB(String&nbsp;url,String&nbsp;user,String&nbsp;password);<BR>　　void&nbsp;close();<BR>}&nbsp;<BR><BR>　　这个接口只定义两个方法，没有任何有实际意义的代码，具体的代码由实作这个接口的类来给出，比如Mysql.java：<BR><BR>Package&nbsp;org.bromon.test;<BR>import&nbsp;java.sql.*;<BR>public&nbsp;class&nbsp;Mysql&nbsp;implements&nbsp;DB<BR>{<BR>　　private&nbsp;String&nbsp;url=”jdbc:mysql:localhost:3306/test”;<BR>　　private&nbsp;String&nbsp;user=”root”;<BR>　　private&nbsp;String&nbsp;password=””;<BR>　　private&nbsp;Connection&nbsp;conn;<BR>　　public&nbsp;Connection&nbsp;openDB(url,user,password)<BR>　　{<BR>　　　　//连接数据库的代码<BR>　　}<BR><BR>　　public&nbsp;void&nbsp;close()<BR>　　{<BR>　　　　//关闭数据库<BR>　　}<BR>}&nbsp;<BR><BR>　　类似的当然还有Oracle.java等等，接口DB给这些类归了个类，在应用程序中我们这样定义对象：<BR><BR>　　org.bromon.test.DB&nbsp;myDB;<BR><BR>　　使用myDB来操作数据库，就可以不用管实际上我所使用的是哪个类，这就是所谓的“开-闭”原则。但是问题在于接口是不能实例化的，myDB=new&nbsp;DB()，这样的代码是绝对错误的，我们只能myDB=new&nbsp;Mysql()或者myDB=new&nbsp;Oracle()。麻烦了，我还是需要指定具体实例化的是哪个类，用了接口跟没用一样。所以我们需要一个工厂：<BR><BR>package&nbsp;org.bromon.test;<BR>public&nbsp;class&nbsp;DBFactory<BR>{<BR>　　public&nbsp;static&nbsp;DB&nbsp;Connection&nbsp;getConn()<BR>　　{<BR>　　　　Return(new&nbsp;Mysql());<BR>　　}<BR>}&nbsp;<BR><BR>　　所以实例化的代码变成：myDB=DBFactory.getConn()；<BR><BR>　　这就是23种模式中最基础的普通工厂(Factory)，工厂类负责具体实例化哪个类，而其他的程序逻辑都是针对DB这个接口进行操作，这就是“针对接口编程”。责任都被推卸给工厂类了，当然你也可以继续定义工厂接口，继续把责任上抛，这就演变成抽象工厂(Abstract&nbsp;Factory)。<BR><BR>　　整个过程中接口不负责任何具体操作，其他的程序要连接数据库的话，只需要构造一个DB对象就OK，而不管工厂类如何变化。这就是接口的意义----抽象。<BR><BR>　　继承的概念不用多说，很好理解。为什么要继承呢？因为你想重用代码？这绝对不是理由，继承的意义也在于抽象，而不是代码重用。如果对象A有一个run()方法，对象B也想有这个方法，所以有人就Class&nbsp;B&nbsp;extends&nbsp;A。这是不经大脑的做法。如果在B中实例化一个A，调用A的Run()方法，是不是可以达到同样的目的？如下：<BR><BR>Class&nbsp;B<BR>{<BR>　　A&nbsp;a=new&nbsp;A();<BR>　　a.run();<BR>}&nbsp;<BR><BR>　　这就是利用类的聚合来重用代码，是委派模式的雏形，是GoF一贯倡导的做法。<BR><BR>　　那么继承的意义何在？其实这是历史原因造成的，最开始的OO语言只有继承，没有接口，所以只能以继承来实现抽象，请一定注意，继承的本意在于抽象，而非代码重用（虽然继承也有这个作用），这是很多Java烂书最严重的错误之一，它们所造成的阴影，我至今还没有完全摆脱，坏书害人啊，尤其是入门类的，流毒太大。什么时候应该使用继承？只在抽象类中使用，其他情况下尽量不使用。抽象类也是不能实例化的，它仅仅提供一个模版而已，这就很能说明问题。<BR><BR>　　软件开发的万恶之源，一是重复代码而不是重用代码，二是烂用继承，尤以c++程序员为甚。Java中取缔多重继承，目的就是制止烂用继承，实是非常明智的做法，不过很多人都不理解。Java能够更好的体现设计，这是让我入迷的原因之一。</FONT><BR><img src ="http://www.blogjava.net/tfc2005/aggbug/6028.html" width = "1" height = "1" /><br><br><div align=right><a style="text-decoration:none;" href="http://www.blogjava.net/tfc2005/" target="_blank">佟福春</a> 2005-06-13 10:32 <a href="http://www.blogjava.net/tfc2005/archive/2005/06/13/6028.html#Feedback" target="_blank" style="text-decoration:none;">发表评论</a></div>]]></description></item></channel></rss>