java继续_Java中消除实现继续和面向接口编程

在匆忙之际理清消除实现继续和面向接口编程这样两个大题目可不是一件轻易的事情,尤其考虑到自身的熟悉水平。坦白的说,这又是一篇“炒冷饭”的文章,但这“冷饭”又确实不好炒。因此,在阅读了这篇文章之后,你可要批判地接受(拒尽)我的观点,尽管我的观点也是来自于别人的观点。

继续是面向对象中很重要的概念。假如考虑到Java语言特性,继续分为两种:接口继续和实现继续。这只是技术层面的题目,即便C++中不存在接口的概念,但它的虚基类实际上也相当于接口。对于OO的初学者来说,他们很希看自己的程序中出现大量的继续,由于这样看起来很OO。但滥用继续会带来很多题目,尽管有时候我们又不得不使用继续解决题目。

相比于接口继续,实现继续的题目要更多,它会带来更多的耦合题目。但接口继续也是有题目的,这是继续本身的题目。实现继续的很多题目出于其自身实现上,因此这里重点讨论实现继续的题目。

举个例子(这个例子实在太老套了)。我要实现一个Stack类,我想当然地选择Stack类继续于ArrayList类(你也可以以为我很想OO些或者出于本性的懒惰);现在又有了新的需求,需要实现一个线程安全的Stack,我又定义了一个ConcurrentStack类继续于Stack并覆盖了Stack中的部分代码。

由于Stack继续于ArrayList,Stack不得不对外暴露出ArrayList所有的public方法,即便其中的某些方法对它可能是不需要的;甚至更糟的是,可能其中的某些方法能改变Stack的状态,而Stack对这些改变并不知情,这就会造成Stack的逻辑错误。

假如我要在ArrayList中添加新的方法,这个方法就有可能在逻辑上破坏它的派生类Stack、

ConcurrentStack。因此在基类(父类)添加方法(修改代码)时,必须检查这些修改是否会对派生类产生影响;假如产生影响的话,就不得不对派生类做进一步的修改。假如类的继续体系不是一个人完成的,或者是修改别人的代码的情况下,很可能由于继续产生难以觉察的BUG。

题目还是有的。我们有时会见到这样的基类,它的一些方法只是抛出异常,这意味着假如派生类支持这个方法就重写它,否则就如父类一样抛出异常表明其不支持这个方法的调用。我们也能见到它的一个变种,父类的方法是抽象的,但不是所有的子类都支持这个方法,不支持的方法就以抛出异常的方式表明态度。这种做法是很不友好和很不安全的,它们只能在运行时被“侥幸捕捉”,而很多漏网的异常方法可能会在某一天忽然出现,让人不知所措。

引起上面题目的很重要的原因便是基类和派生类之间的耦合。往往只是对基类做了小小的改动,却不得不重构它们的所有的派生类,这就是臭名昭著的“脆弱的基类”题目。由于类之间的关系是存在的,因此耦合是不可避免的甚至是必要的。但在做OO设计时,当碰到如基类和派生类之间的强耦合关系,我们就要思量思量,是否一定需要继续呢?是否会有其他的更优雅的替换方案呢?假如一定要学究的话,你会在很多书中会看到这样的原则:假如两个类之间是IS-A关系,那么就使用继续;假如两个类之间是Has-A的关系,那么就使用委派。很多时候这条原则是适用的,但IS-A并不能做为使用继续的尽对理由。有时为了消除耦合带来的题目,使用委派等方法会更好地封装实现细节。继续有时会对外及向下暴露太多的信息,在GOF的设计模式中,有很多模式的目的就是为了消除继续。

关于何时采用继续,一个重要的原则是确定方法是否能够共享。比如DAO ,可以将通用的CRUD 方法定在一个抽象DAO

中,具体的DAO 都派生自这个抽象类。严格的说,抽象DAO 和派生的DAO 实现并不具有IS -A

关系,我们只是为了避免重复的方法定义和实现而作出了这一技术上的选择。可以说,使用接口还是抽象类的原则是,假如多个派生类的方法内容没有共同的地方,就用接口作为抽象;假如

多个派生类

的方法含有共同的地方,就用抽象类作为抽象。当这一原则不适用于接口继续,假如出现接口继续,就会相应地有实现继续(基类更多的是抽象类)。

现在说说面向接口编程。在众多的灵敏方法中,面向接口编程总是被大师们反复的夸大。面向接口编程,实际上是面向抽象编程,将抽象概念和具体实现相隔离。这一原则使得我们拥有了更高层次的抽象模型,在面对不断变更的需求时,只要抽象模型做的好,修改代码就要轻易的多。但面向接口编程不意味着非得一个接口对应一个类,过多的不必要的接口也可能带来更多的工作量和维护上的困难。

相比于继续,OO中多态的概念要更重要。一个接口可以对应多个实现类,对于声明为接口类型的方法参数、类的字段,它们要比实现类更易于扩展、稳定,这也是多态的优点。假如我以实现类作为方法参数定义了一个方法void

doSomething(ArrayList list),但假如领导哪天觉得

ArrayList不如LinkedList更好用,我将不得不将方法重构为void doSomething(LinkedList

list),相应地要在所有调用此方法的地方修改参数类型(很遗憾地,我连对象创建也是采用ArrayList list = new

ArrayList()方式,这将大大增加我的修改工作量)。假如领导又觉得用list存储数据不如set好的话,我将再一次重构方法,但这一次我变聪明了,我将方法定义为void

doSomething(Set set),创建对象的方式改为Set set = new

HashSet()。但这样仍不够,假如领导又要求将set改回list怎么办?所以我应该将方法重构为void

doSomething(Collection collection),

Collection的抽象程度最高,更易于替换具体的实现类。即便需要List或者Set固有的特性,我也可以做向下类型转换解决题目,尽管这样做并不安全。

面向接口编程最重要的价值在于隐躲实现,将抽象的实现细节封装起来而不对外开放,封装这对于Java EE

中的分层设计和框架设计尤其重要。但即便在编程时使用了接口,我们也需要将接口和实现对应起来,这就引出如何创建对象的题目。在创建型设计模式中,单例、工厂方法(模板方法)、抽象工厂等模式都是很好的解决办法。现在流行的控制反转(也叫依靠注进)模式是以声明的方式将抽象与实现连接起来,这既减少了单调的工厂类也更易于单元测试。

做个总结吧。尽管我竭力批驳继续的不好鼓吹接口的好,但这并不是尽对的。滥用继续、滥用接口都会带来题目。做Java

EE开发的很多朋友抱怨DAO、Service中一个接口一个类的实现方式,尽管它们似乎看起来已成为业界的最佳实践之一。也许排除掉接口会使程序更“瘦”一些,但“瘦”并一定就“好”,需要根据项目的具体情况而定。关于继续和接口的最佳实践,各位看官还是需要自身的经验积累和总结了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/575771.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

《转》VC++多线程编程

原地址:http://www.cnblogs.com/wxfasdic/archive/2010/09/23/1833522.html留个纪念,不错的总结。十个例子清晰列举啦多线程编程的奥妙。 VC中多线程使用比较广泛而且实用,在网上看到的教程.感觉写的挺好. 一、问题的提出编写一个耗时的单线程程序&#…

array函数参数 scala_scala – 在Spark SQL中将数组作为UDF参数传递

很可能不是最漂亮的解决方案,但你可以尝试这样的事情:def getCategory(categories: Array[String]) {udf((input:String) > categories(input.toInt))}df.withColumn("newCategory", getCategory(myArray)(col("myInput")))您还可以尝试一系…

Java数据类型BooleanDemo

转载于:https://www.cnblogs.com/suncoolcat/p/3320306.html

beetle.java 分析_Beetl模板引擎入门教程

最近项目中有个邮件发送的需求,不过要求发送的HTML格式的邮件。由于Beetl对java语言的良好支持和很好的性能,我们决定使用Beetl作为我们的模板引擎。Beetl官网已经有了很详细的教程,所以本篇侧重于实战应用,适合需要不懂beetl或其…

WebScoket 规范 + WebSocket 协议

WebSocket握手协议 1、客户端握手请求(注意:键值之间有一个空格,行间有换行符号0x13x10或者说\r\n) GET /WebSocket/LiveVideo HTTP/1.1 Upgrade: WebSocket Connection: Upgrade Host: localhost:8080 (客户端请求主机) Origin:…

heap python_数据结构-堆(Heap) Python实现

堆(Heap)可以看成近似完全二叉树的数组,树中每个节点对应数组中一个元素。除了最底层之外,该树是完全充满的,最底层是从左到右填充的。堆包括最大堆和最小堆:最大堆的每一个节点(除了根结点)的值不大于其父节点;最小堆…

多个 App 间启动

http://developer.nokia.com/: URI associations for Windows Phone http://msdn.microsoft.com/: Auto-launching apps using file and URI associations for Windows Phone 8 代码示例转载于:https://www.cnblogs.com/sirkevin/p/3325035.html

im4java 文档_im4java学习---阅读documentation文档

Utilities----im提供的一些工具类①、读取图片文件信息---Info类我们之前的做法:op.format("width:%w,height:%h,path:%d%f,size:%b%[EXIF:DateTimeOriginal]");IdentifyCmd identifyCmd new IdentifyCmd(useGM);使用工具类Info:Info imageIn…

函数体中定义的结构体和类型

源代码&#xff1a; 1 #include <stdio.h>2 struct smonth // point 13 {4 int a;5 int b;6 };7 8 int func1()9 { 10 struct smonth{ 11 int a; 12 int b; 13 }; 14 15 ty…

java listview用法_Java ListView.setMultiChoiceModeListener方法代码示例

import android.widget.ListView; //导入方法依赖的package包/类Overridepublic void onActivityCreated(Nullable Bundle savedInstanceState) {super.onActivityCreated(savedInstanceState);lAdapter new LabelAdapter(getActivity(), null, 0);setListAdapter(lAdapter);g…

预编译指令与宏定义

#if #elif [defined(), !defined()] #else #ifdef #ifndef #endif // 条件编译 /* 头文件防止多次被包含 */ #ifndef ZLIB_H #define ZLIB_H#endif /* ZLIB_H *//* 用C方式来修饰函数与变量 */ #ifdef __cplusplus extern "C" { #endif int add(int a, …

java mock server_java – 使用MockRestServiceServer模拟REST调用

我正在尝试编写一个JUnit测试用例,用于测试辅助类中的方法.该方法使用REST调用外部应用程序,这是我试图在JUnit测试中模拟的调用.辅助方法使用Spring的RestTemplate进行REST调用.在我的测试中,我创建了一个模拟REST服务器并模拟REST模板并将它们实例化为&#xff1a;Beforepubl…

BZOJ 2597 剪刀石头布(最小费用最大流)(WC2007)

Description 在一些一对一游戏的比赛&#xff08;如下棋、乒乓球和羽毛球的单打&#xff09;中&#xff0c;我们经常会遇到A胜过B&#xff0c;B胜过C而C又胜过A的有趣情况&#xff0c;不妨形象的称之为剪刀石头布情况。有的时候&#xff0c;无聊的人们会津津乐道于统计有多少这…

java swt最小化到托盘_SWT 中实现最小化到托盘图标,并只能通过托盘的弹出菜单关闭程序...

package com.unmi;import org.eclipse.swt.*;import org.eclipse.swt.events.*;import org.eclipse.swt.graphics.*;import org.eclipse.swt.widgets.*;/*** SWT 3.0 开始引入了 Tray&#xff0c;可以在系统栏放置你的程序图标了* 本程序实现的功能有四&#xff1a;* 1. 点击窗…

HDU 1476 Sudoku Killer

Sudoku Killer Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 3367 Accepted Submission(s): 1063 Problem Description自从2006年3月10日至11日的首届数独世界锦标赛以后&#xff0c;数独这项游戏越来越受到…

java .net 互通redis_C# servicestack.redis 互通 java jedis

本文是基于jedis的一致性环哈希来修改的&#xff0c;.net选的是servicestack.redis组件来修改无奈两个组件都有各自的一致性环哈希算法&#xff0c;不兼容&#xff0c;那就选一个作为标准&#xff0c;修改另一个咯。本文选择jedis的一致性环哈希作为标准&#xff0c;进而修改.n…

java 多重压缩下载_Java 多文件边压缩边下载

有时我们希望在后台实时生成文件并下载到客户端GetMapping(value "download")public void download(HttpServletResponse response) {try(OutputStream outputStream response.getOutputStream();ZipOutputStream zipOutputStream new ZipOutputStream(outputStre…

wdatepicker使用指南

wdatepicker使用指南 http://blog.csdn.net/zengxin2008/article/details/7248964#t63posted on 2013-09-22 15:45 moonfans 阅读(...) 评论(...) 编辑 收藏 转载于:https://www.cnblogs.com/moonfans/p/3333251.html

纵表、横表互转的SQL

纵表、横表互转的SQL By&#xff1a;大志若愚 1、建表&#xff1a; 纵表结构 Table_A create table Table_A (姓名 varchar(20),课程 varchar(20),成绩 int ) insert into Table_A(姓名,课程,成绩) values(张三,语文,60) insert into Table_A(姓名,课程,成绩) values(张三,数学…

java按照商品价格排序_按照指定的类型排序

比如有三本书&#xff0c;书名和价格分别如下&#xff1a;《猎物》18.4元、《大明王朝》59.0元、《快乐密码》25.0元。前几天的一篇文章中讲到按照中文拼音进行排序&#xff0c;这里如果想按照价格排序的话怎么办&#xff1f;其实这里就牵涉到在Java中怎样对对象进行排序处理&a…