java 接口学习

你应该知道接口是一种契约,它与实现方式无关但是类,即使是抽象类,你都能自定义成员变量,而成员变量往往就与实现方式有关。这一点的实际意义不大。但是有一点,类会暴露太多不必要,甚至不能暴露的东西,你

大家好,又见面了,我是全栈君,今天给大家准备了Idea注册码。

你应该知道接口是一种契约,它与实现方式无关

但是类,即使是抽象类,你都能自定义成员变量,而成员变量往往就与实现方式有关。

这一点的实际意义不大。

但是有一点,类会暴露太多不必要,甚至不能暴露的东西,你看一下java.util中,大部分的数据结构,都被设计成了接口-抽象类-最后实际类

例如Collection-List
L-AbstractCollection
L-AbstractList
L-ArrayList

但是有一个,由于历史原因,被设计成了类,比如Stack extends Vector,
你应该知道Stack的数据访问模式,就只能是LIFO,但是Vector是一个List,可以随机访问,可以任意增减。

结果Stack s = new Stack();

不光能够pop/push/peer还能add,get,set,remove

如果你用一个接口IStack 里面只有pop/push/peer方法,然后仍然通过继承Vector的方式实现,

IStack s = new MyStack();

此时,就无法add,get,set,remove

举个生动一点的例子

public interface BritishSpy {
public String speak(); //英国间谍讲英语
}

public interface GermanSpy {
public String sprechen(); //德国间谍讲德语
}

public class DoubleAgent implements BritishSpy, GermanSpy {
public String speak() { return "Hello"; }
public String sprechen() { return "Gutentag"; }
}
// hovertree.com
public class Agency {
public static void toMI5(BritishSpy spy) {
//军情5处当然只能说英语,做英国间谍
spy.speak();
//spy.sprechen();不可见
}

public static void inGermany(GermanSpy spy) {
//spy.speak();不可见
spy.sprechen();
}

public static void main(String[] args) {
DoubleAgent da = new DoubleAgent();
BritishSpy es = (BritishSpy) da;
GermanSpy gs = (GermanSpy) da;
toMI5(da); //MI5也不知道他是一个双重间谍,只知道他是BritishSpy
toMI5(es); //更安全  何问起
//toMI5(gs); 不可能
inGermany(da); //在德国还是安全的,德国人不知道他的双重间谍身份,只知道他是GermanSpy
inGermany(gs); 
//inGermany(es); 不可能
}
}

 

假设你只用class,因为不能多重继承,所以,speak()/sprechen()比然声明在同一个class里面

public abstract class DoubleAgent extends Spy/**(略…)*/ {
public abstract String speak();
public abstract String sprechen();
}

public class PoorDoubleAgent {
public String speak() { return “Hello”; }
public String sprechen() { return “Gutentag”; }
}

晚了,不管你PoorDoubleAgent a = new PoorDoubleAgent();还是DoubleAgent a = new PoorDoubleAgent();,全世界都知道他是一个双重间谍,他到哪里都必死无疑
前面举了一个关于“安全性”方面的例子

接口只暴露给对方(比如Agent的军情5处方法)它所需要的足够信息,其他无关的,甚至有害的信息不会暴露给对方。因为,我传给你的是接口类型,我除了是这个接口(和这个接口的父接口,inteface A extends B, C)的实例外,你顶多知道我是一个Object(不是int:P),其他的姓甚名谁,哪里住址,父母安好,兄妹几何都与你无关,我们只需要关心我们签订的合同(接口) 

再举一个有关灵活性方面的例子

假设某公司已经有一个更新过N代的,逻辑复杂无比
public class A extends B /** where b extends c, c extends d and so on… */ {
public void init() {…}
public void release() {…}
public String doXXX() {…}
public String doYYY() {…}
}

而这个A又被很多类继承或使用,doXXX/doYYY 方法已经无法更改

假设现在这个公司要参加某个标准化组织,而这个组织要求所有提供这样的方法

String getXXX(); String getYYY();

加入用接口标准化组织只要规定成员们都实现
public interface IBusiness {
String getXXX();
String getYYY();
}
而这个公司只需要稍微改写一点点即可
public class A extends B /** where b extends c, c extends d and so on… */
implements IBusiness {
public String getXXX() { return doXXX(); }
public String getYYY() { return doYYY(); }

//保留
public void init() {…}
public void release() {…}
public String doXXX() {…}
public String doYYY() {…}
}

这样既满足了标准化的要求,又满足了无需修改原来继承A或者使用A的无数个class(有些可能在用户那里,不可能更改)

假如不用接口,你有两个选择:数典忘祖或者自绝于人

数典忘祖:
你的新A必须继承标准化组织的Business,原来a,b, c d…里面的代码全部得重新组织到这个新的A里面,与此同时,那些调用或者继承A的class难保不需要重写

自绝于人
原来的就让它去,谁也别再提它了,我们以后就用新的NewA,结果,你的新客户倒是满足了标准化,你的老客户可就 :< :@ :$,而且以后需要维护A和NewA
定义接口:
interface Fight{
void fight();
}

肥肥和瘦瘦去实现这个接口:
class FatFat implements Fight{
public void fight(){
System.out.println(“FatFat 打人很痛!”);
}
}

class ThinThin implements Fight{
public void fight(){
System.out.println(“ThinThin 打人一点都不痛!!哈哈。”);
}
}

然后你可能会这另一个类中使用到FatFat和ThinThin的对象,然后都去执行fight,但是你可能不到运行时就不会知道是具体的那个类的对象,这是你就感谢他们都实现了Fight接口,你可以向上转型,然后通过运行时的多态产生你想要的行为。http://hovertree.com/menu/java/

Fight a=new FatFat();
Fight b=new ThinThin();
a.fight();
b.fight();
这样就会执行不同的动作。

或者如果你有一个方法
f(Fight i){
i.fight();
}

如果c是实现了Fight接口的其中一个类,那么你就可以这样使用这个方法:
f(c);
你不需要知道c究竟是什么对象(不管是FatFat还是ThinThin),你都可以得到你想要的fight动作

转自:http://hovertree.com/h/bjaf/0c6sbcn8.htm

推荐:http://www.cnblogs.com/roucheng/p/3504465.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/120590.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • Keil5下载安装教程并完成注册(配图操作)[通俗易懂]

    Keil5下载安装教程并完成注册(配图操作)[通俗易懂]Keil5安装教程以及安装包下载1、安装包下载2、下载并解压安装包,并按步骤完成安装2.1、运行安装程序,点击next2.2、勾选accept,点击next2.3、选择安装路径,点击next2.4、信息随意填写,点击next2.5、等待安装2.6、点击finish,完成安装3、打开注册机,完成注册3.1、以管理员身份运行keil53.2、点击licence3.3、打开注册机3.4、复制CID,选择arm,点击Generate3.5、复制生成的4步骤到keil5,点击ADD3.6、注册成功1、安装包下载微

  • 文科生也能学会的Excel VBA 宏编程入门

    文科生也能学会的Excel VBA 宏编程入门文章目录VBA宏编程简介准备工作打开宏功能打开“开发工具”选项卡VBA编程示例任务介绍文件宏全局宏VBA宏编程简介VBA编程是Office系列软件自带的编程功能,也就是说不只是Excel,Word和PPT也能进行VBA编程。而“宏”可以理解为一组自动化程序,执行一个宏就能执行其对应的一系列操作。创建宏主要有两种方法,一种是录制宏,也就是将人的一些操作录下来,需要的时候执行宏就可以自动重复这些操…

  • RedisClient 可视化Redis工具下载「建议收藏」

    RedisClient 可视化Redis工具下载「建议收藏」https://github.com/caoxinyu/RedisClient网站下边的文档说明中如果电脑没有JDK,则点redisclient-win32.x86.2.0.exe如果有JDK,若是64位系统则点redisclient-win32.x86_64.2.0.jar32位系统点redisclient-win32.x86.2.0.jar…

    2022年10月12日
  • 最新、最全、最准确的手机号正则表达式

    最新、最全、最准确的手机号正则表达式更新到2018年1月,支持最新的166号段/***手机号码正则表达式*/privatestaticfinalStringPHONE_NUMBER_REG="^(13[0-9]|14[579]|15[0-3,5-9]|16[6]|17[0135678]|18[0-9]|19[89])\\d{8}$";

  • latex中bibtex生成参考文献_英文参考文献自动生成方法

    latex中bibtex生成参考文献_英文参考文献自动生成方法创建BibTeX文件  BibTex是一种专门用于协调LaTeX的参考文献处理的文件格式,其后缀为.bib。BibTex可以用JabRef,bibtex等软件生成,也可以直接用记事本创建,操作起来非常灵活,这里主要介绍用JabRef来生成BibTex文件。JabRef是一套开放源代码、处理BibTeX格式的文献管理软件,提供了简易操作的界面来编辑BibTeX档案,功能包含从网络上的科学数据库汇…

  • linux网络配置出现E325,打开Vi编辑器出现E325: ATTENTION的解决方法

    linux网络配置出现E325,打开Vi编辑器出现E325: ATTENTION的解决方法当打开Vi编辑器时出现以下的提示时不要着急E325:ATTENTIONFoundaswapfilebythename”/etc/.profile.swp”ownedby:rootdated:WedJul1410:01:102010filename:/etc/profilemodified:YESusername:roothostname:zh…

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号