Java中的回调机制,这篇给你整的明明白白的,一招让你拿下seata分布式事务框架

回调的思想是:

  • 类A的a()方法调用类B的b()方法

  • 类B的b()方法执行完毕主动调用类A的callback()方法

通俗而言: 就是A类中调用B类中的某个方法C, 然后B类中反过来调用A类中的方法D, D这个方法就叫回调方法, 这样子说你是不是有点晕晕的, 其实我刚开始也是这样不理解, 看了人家说比较经典的回调方式:

  1. class A实现接口CallBack callback——背景1

  2. class A中包含一个class B的引用b ——背景2

  3. class B有一个参数为callback的方法f(CallBack callback) ——背景3

  4. A的对象a调用B的方法 f(CallBack callback) ——A类调用B类的某个方法 C

  5. 然后b就可以在f(CallBack callback)方法中调用A的方法 ——B类调用A类的某个方法D

回调的种类

=====

回调分为同步回调和异步回调, 假如以买彩票的场景来模拟, 我买彩票, 调用彩票网,给我返回的结果确定是否中奖,同步回调就是,我买了彩票之后, 需要等待彩票网给我返回的结果, 这个时候我不能做其他事情, 我必须等待这个结果, 这就叫同步回调, 同步, 就意味着等待, 我不能去做其他事情, 必须等待。

异步回调就是, 我买了彩票之后, 可以去做其他事情, 然后当彩票网有了结果和消息, 再给我返回消息, 其中最明显的方式就是在得到彩票结果的函数之中, 添加一个其他的方法, 如果我的其他方法可以立即执行, 那么就是异步的(给出是否中奖需要花费很长的时间), 而在测试函数之中, 前后两个, 那是发生在测试函数的线程之中的, 肯定是一前一后按照次序的, 在这个地方不是显示同步异步的地点.

同步回调

====

同步回调和异步回调, 主要体现在其是否需要等待. 同步调用, 如果被调用一方的APi(第三方API), 处理问题需要花很长时间, 我们需要等待, 那就是同步回调, 如果调用完之后不需要理解得到结果, 我们调完就走, 去做其他事情, 那就是异步调用, 异步调用需要在我们调用第三方API处, 开启一个新的线程即可, 而同步调用和平常的调用没有任何区别.

例子

==

OrderResult接口, 其中的方法getOrderResult

public interface OrderResult {

/**

* 订购货物的状态

* @param state

* @return

*/

//参数可以不用, 用不用按照自己的实际需求决定

public String getOrderResult(String state);

}

Store类, 商店提供会无预定消息返回的接口, 回调OrderResult接口的方法, 给其返回预订商品的状态, 重点是returnOrderGoodsInfo(OrderResult order)方法, 体现了回调的回. Store是被调用的一方, 被调用的一方, 要回过去调用调用一方的方法, 这个方法实际上是回调接口的方法.

public class Store {

@Getter

@Setter

private String name;

Store(String name) {

this.name = name;

}

/回调函数, 将结构传给那个我们不能直接调用的方法, 然后获取结果/

public String returnOrderGoodsInfo(OrderResult order) {

String[] s = {“订购中…”, “订购失败”, “即将发货!”, “运输途中…”, “已在投递”};

Random random = new Random();

int temp = random.nextInt(5);

String s1 = s[temp];

return order.getOrderResult(s1);

}

}

SyncBuyer类, 同步顾客类, 其中获取商品的订购状态,orderGoods(), 调用了store返回商品调用信息的returnOrderGoodsInfo()方法, 但是在Store类的returnOrderGoodsInfo()方法之中, 以OrderResult接口为参数, 反过来调用了OrderResult接口, 相当于调用了其子类SyncBuyer本身, 以它为参数, 调用了getOrderResult(String state)方法, 也就是OrderResult接口的方法, 相当于就完成了一个调用的循环, 然后取到了我们自己无法给出的结果.

这个地方的"循环", 是回调的关键所在, 需要正常调用其他外接提供方法来获取结果的一方, 集成一个回调接口, 实现它, 然后调用第三方的API方法, 第三方在我们调用的方法之中, 以回调结构为参数, 然后调用了接口中的方法, 其中可以返回相应的结果给我们.

需要说明的是, 我们虽然实现了这个接口的方法, 但是我们自己的类之中, 或者说此类本身, 却没法调用这个方法, 也可以说, 此类调用这个方法是不会产生有效的结果的. 回调的回, 就体现在此处, 在Store类之中的returnOrderGoodsInfo(OrderResult order)方法之中, 得到了很好的体现.

/同步, 顾客在商店预订商品, 商店通知顾客预订情况/

public class SyncBuyer implements OrderResult {

@Getter

@Setter

private Store store;//商店

@Getter

@Setter

private String buyerName;//购物者名

@Getter

@Setter

private String goodsName;//所购商品名

SyncBuyer(Store store, String buyerName, String goodsName) {

this.store = store;

this.buyerName = buyerName;

this.goodsName = goodsName;

}

/调用从商店返回订购物品的信息/

public String orderGoods() {

String goodsState = store.returnOrderGoodsInfo(this);

System.out.println(goodsState);

myFeeling();// 测试同步还是异步, 同步需要等待, 异步无需等待

return goodsState;

}

public void myFeeling() {

String[] s = {“有点小激动”, “很期待!”, “希望是个好货!”};

Random random = new Random();

int temp = random.nextInt(3);

System.out.println(“我是” + this.getBuyerName() + ", 我现在的感觉: " + s[temp]);

}

/*被回调的方法, 我们自己不去调用, 这个方法给

《一线大厂Java面试题解析+后端开发学习笔记+最新架构讲解视频+实战项目源码讲义》

【docs.qq.com/doc/DSmxTbFJ1cmN1R2dB】 完整内容开源分享

出的结果, 是其他接口或者程序给我们的, 我们自己无法产生*/

@Override

public String getOrderResult(String state) {

return “在” + this.getStore().getName() + “商店订购的” + this.getGoodsName() + "玩具, 目前的预订状态是: " + state;

}

}

Test2Callback类, 测试同步回调的结果,

public class Test2Callback {

public static void main(String[] args) {

Store wallMart = new Store(“沙中路沃尔玛”);

SyncBuyer syncBuyer = new SyncBuyer(wallMart, “小明”, “超能铁扇公主”);

System.out.println(syncBuyer.orderGoods());

}

}

异步回调

====

同步回调和异步回调的代码层面的差别就是是否在我们调用第三方的API处, 为其开辟一条新的线程, 其他并无差异。Java知音公众号内回复”面试题聚合“,送你一份面试题宝典

例子

==

OrderResult接口, 其中的方法getOrderResult

public interface OrderResult {

/**

* 订购货物的状态

* @param state

* @return

*/

//参数可以不用, 用不用按照自己的实际需求决定

public String getOrderResult(String state);

}

Store类, 商店提供会无预定消息返回的接口, 回调OrderResult接口的方法, 给其返回预订商品的状态.

public class Store {

@Getter

@Setter

private String name;

Store(String name) {

this.name = name;

}

/回调函数, 将结构传给那个我们不能直接调用的方法, 然后获取结果/

public String returnOrderGoodsInfo(OrderResult order) {

String[] s = {“订购中…”, “订购失败”, “即将发货!”, “运输途中…”, “已在投递”};

Random random = new Random();

int temp = random.nextInt(5);

String s1 = s[temp];

return order.getOrderResult(s1);

}

}

NoSyncBuyer类, 异步调用Store类的returnOrderGoodsInfo(OrderResult order)方法, 来返回商品转改的结果.

/异步/

@Slf4j

public class NoSyncBuyer implements OrderResult {

@Getter

@Setter

private Store store;//商店

@Getter

@Setter

private String buyerName;//购物者名

@Getter

@Setter

private String goodsName;//所购商品名

NoSyncBuyer(Store store, String buyerName, String goodsName) {

this.store = store;

this.buyerName = buyerName;

this.goodsName = goodsName;

}

/调用从商店返回订购物品的信息/

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值