Java为什么要用接口回调调用被实现的方法,而不直接用实现该接口的类创建对象使用呢?

Java为什么要用接口回调调用被实现的方法,而不直接用实现该接口的类创建对象使用呢?

作者:饺克力
链接:https://www.zhihu.com/question/52473186/answer/974133599
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

回调,顾名思义信息回传的意思,比如A让B做了一件事,做完后B要回传给A一个信息,或者说在调用A一个方法,这就是回调。那么为什么JAVA推荐通过接口的方式来实现回调呢,通过以下几个场景逐步来理解:
首先,没有任何回调的

场景1:Boss让员工Worker做了一件事情,因为Boss要调用员工的方法,因此Boss需要持有员工Worker的引用:

class test  
{
	public static void main (String[] args) throws java.lang.Exception
	{
		Boss wbz = new Boss();
		Worker worker1 = new Worker();
		wbz.anpai(worker1);
	}
}

class Boss
{
    public void anpai(Worker Worker){
        Worker.dojob();
    }
}

class Worker
{
    public void dojob () {
        System.out.println("working...");
    }
}


场景2:用引用的方法加入回调:即给Boss加入一个新方法:workerHasFinished,即员工Worker完成工作后,可以调用Boss的这个方法,Boss也就可以继续执行后续自己的任务了。因此Worker要调用Boss里的方法时,那么首先想到的就是让Worker也反过来持有Boss的引用,这是合情合理的。因此,在Boss调用Woker方法的时候,需要把自己(this)作为参数传给Worker。这样写,功能是完全可以实现的。如下代码:


class test  
{
	public static void main (String[] args) throws java.lang.Exception
	{
		Boss wbz = new Boss();
		Worker worker1 = new Worker();
		wbz.anpai(worker1);
	}
}

class Boss
{
    protected int coins = 10;
    public void anpai(Worker Worker){
        Worker.dojob(this);
    }
    
    public void workerHasFinished(){
        System.out.println("I know my worker has finished his job!");
    }
}

class Worker
{
    
    public void dojob (Boss boss) {
        System.out.println("working...");
        boss.workerHasFinished();
    }
}

但这是最优的写法吗?不是的!注意看,作为Boss,突然有一个小金库属性,coins=10,我们只想让员工干活,但是不想让员工在干活的时候,知道自己的其他属性,比如coins,他会发现原来自己老板这么穷。这是非常不安全的。所以作为员工,Worker试着干完活以后,用boss.coins访问了一下,发现竟然可以访问得到boss的这个属性!


class test  
{
	public static void main (String[] args) throws java.lang.Exception
	{
		Boss wbz = new Boss();
		Worker worker1 = new Worker();
		wbz.anpai(worker1);
	}
}

class Boss
{
    protected int coins = 10;
    public void anpai(Worker Worker){
        Worker.dojob(this);
    }
    
    public void workerHasFinished(){
        System.out.println("I know my worker has finished his job!");
    }
}

class Worker
{
    
    public void dojob (Boss boss) {
        System.out.println("working...");
        boss.workerHasFinished();
        System.out.println("And I know my boss has coins = "+boss.coins);
    }
}

这下老板就苦恼了,我们最开始想要的是“做完任务就告诉我”这一件事,但是这件事的代价是帮我干活儿的员工能访问我的属性,那么是否可以用什么方法仅仅实现这一件事的功能,而不用牵一发而动全身呢。我们想到了接口~我们加入一个接口,代码改成如下:

作者:饺克力
链接:https://www.zhihu.com/question/52473186/answer/974133599
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

class test  
{
	public static void main (String[] args) throws java.lang.Exception
	{
		Boss wbz = new Boss();
		Worker worker1 = new Worker();
		wbz.anpai(worker1);
	}
}

interface Huidiao {
    public void toldme();
}

class Boss implements Huidiao
{
    public int coins = 10;
    public void anpai(Worker worker){
        worker.dojob(this);
    }
    
    public void workerHasFinished(){
        System.out.println("I know my worker has finished his job!");
    }
    
    public void toldme(){
        System.out.println("I know my worker has finished his job!");
    }
}

class Worker
{
    public void dojob (Huidiao boss) {
        System.out.println("working...");
        //boss.workerHasFinished();
        boss.toldme();
        //System.out.println("And I know my boss has coins = "+boss.coins);
    }
}

执行结果如下:

working...
I know my worker has finished his job!

Huidiao接口就一个方法,toldme,就是做完了之后,老板要执行的方法。所以Boss实现这个接口,在接口中继续他的工作。对于员工Worker而言,他要调用老板的toldme方法,这时候,不需要Boss的引用,而只需要这个接口的引用就可以了。同时我们测试一下,Boss自身的coins属性,他也获取不到了,同时Boss原本的workerHasFinished方法,他也不能再调用了。(两行都被注释掉了)这也是接口的特性:①实现接口,就要实现(重写)接口里所有的方法。②调用接口,就只能调用实现了这个接口的类的,接口里指定的方法。另外,如果再有另外一个老板Boss2想让Worker来干活,那么他只需要再实现Huidiao这个接口,然后获取worker的引用,即可,worker干完了活儿,只能按照接口中指定的规范来调用他应该回调的函数,而没有权限访问Boss2里任何其他的东西了。

  • 7
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值