Java异步调用转同步的5种方式
1、异步和同步的概念
 同步调用:调用方在调用过程中,持续等待返回结果。
 异步调用:调用方在调用过程中,不直接等待返回结果,而是执行其他任务,结果返回形式通常为回调函数。
2 、异步转为同步的概率
  需要在异步调用过程中,持续阻塞至获得调用结果。
3、异步调用转同步的5种方式
1、使用wait和notify方法
2、使用条件锁
3、Future
4、使用CountDownLatch
5、使用CyclicBarrier
4、构造一个异步调用模型。
我们主要关心call方法,这个方法接收了一个demo参数,并且开启了一个线程,在线程中执行具体的任务,并利用demo的callback方法进行回调函数的调用。大家注意到了这里的返回结果就是一个[0,10)的长整型,并且结果是几,就让线程sleep多久——这主要是为了更好地观察实验结果,模拟异步调用过程中的处理时间。至于futureCall和shutdown方法,以及线程池tp都是为了demo3利用Future来实现做准备的。
public class AsyncCall {
private Random random = new Random(System.currentTimeMillis());
 private ExecutorService tp = Executors.newSingleThreadExecutor();    //demo1,2,4,5调用方法
    public void call(BaseDemo demo){        
                new Thread(()->{            long res = random.nextInt(10);            try {
                Thread.sleep(res*1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            demo.callback(res);
        }).start();
    }    //demo3调用方法
    public Future<Long> futureCall(){
            return tp.submit(()-> {
             long res = random.nextInt(10);            try {
             Thread.sleep(res*1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }            return res;
        });
    }    public void shutdown(){
        tp.shutdown();
    }
}
demo的基类:
public abstract class BaseDemo {
      protected AsyncCall asyncCall = new AsyncCall();
      public abstract void callback(long response);
      public void call(){
        System.out.println("发起调用");
        asyncCall.call(this);
        System.out.println("调用返回");
    }
}
5、各种方法的具体实现
5.1、使用wait和notify方法
可以看到在发起调用后,主线程利用wait进行阻塞,等待回调中调用notify或者notifyAll方法来进行唤醒。注意,和大家认知的一样,这里wait和notify都是需要先获得对象的锁的。在主线程中最后我们打印了一个内容,这也是用来验证实验结果的,如果没有wait和notify,主线程内容会紧随调用内容立刻打印;而像我们上面的代码,主线程内容会一直等待回调函数调用结束才会进行打印。
没有使用同步操作的情况下,打印结果:
public class Demo1 extends BaseDemo{    private final Object lock = new Object();    @Override
      public void callback(long response) {
        System.out.println("得到结果");
        System.out.println(response);
        System.out.println("调用结束");        synchronized (lock) {
            lock.notifyAll();
        }
    }    public static void main(String[] args) {
        Demo1 demo1 = new Demo1();
        demo1.call();        synchronized (demo1.lock){            try {
                demo1.lock.wait();
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
        System.out.println("主线程内容");
    }
}
没有使用同步操作的情况下,打印结果:
发起调用
调用返回
主线程内容
得到结果
1
调用结束
而使用了同步操作后:
发起调用
调用返回
得到结果
9
调用结束
主线程内容
5.2、使用条件锁
本上和方法5.2没什么区别,只是这里使用了条件锁,两者的锁机制有所不同。
public class Demo2 extends BaseDemo {
  private final Lock lock = new ReentrantLock();
private final Condition con = lock.newCondition();
@Override
    public void callback(long response) {
        System.out.println("得到结果");
        System.out.println(response);
        System.out.println("调用结束");
        lock.lock();        try {
            con.signal();
        }finally {
            lock.unlock();
        }
    }    public static void main(String[] args) {
        Demo2 demo2 = new Demo2();
        demo2.call();
        demo2.lock.lock();        try {
            demo2.con.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }finally {
            demo2.lock.unlock();
        }
        System.out.println("主线程内容");
    }
}
5.3、Future
使用Future的方法和之前不太一样,我们调用的异步方法也不一样
public class Demo3{
    private AsyncCall asyncCall = new AsyncCall();
    public Future<Long> call(){
        Future<Long> future = asyncCall.futureCall();
        asyncCall.shutdown();        return future;
    }    public static void main(String[] args) {
        Demo3 demo3 = new Demo3();
        System.out.println("发起调用");
        Future<Long> future = demo3.call();
        System.out.println("返回结果");
  while (!future.isDone() && !future.isCancelled());
  try {
            System.out.println(future.get());
        } catch (InterruptedException e)
         {
            e.printStackTrace();
        } catch (ExecutionException e) {
            e.printStackTrace();
        }
        System.out.println("主线程内容");
    }
}
5.4、CountDownLatch
使用CountDownLatch或许是日常编程中最常见的一种了,也感觉是相对优雅的一种:
public class Demo4 extends BaseDemo{
   private final CountDownLatch countDownLatch = new CountDownLatch(1);
   @Override
    public void callback(long response) {
        System.out.println("得到结果");
        System.out.println(response);
        System.out.println("调用结束");
        countDownLatch.countDown();
    }    public static void main(String[] args) {
        Demo4 demo4 = new Demo4();
        demo4.call();        try {
            demo4.countDownLatch.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("主线程内容");
    }
正如大家平时使用的那样,此处在主线程中利用CountDownLatch的await方法进行阻塞,在回调中利用countDown方法来使得其他线程await的部分得以继续运行。
当然,这里和demo1和demo2中都一样,主线程中阻塞的部分,都可以设置一个超时时间,超时后可以不再阻塞
5.5、CyclicBarrier
CyclicBarrier的情况和CountDownLatch有些类似:
public class Demo5 extends BaseDemo{
 private CyclicBarrier cyclicBarrier = new CyclicBarrier(2);
   @Override
    public void callback(long response) {
        System.out.println("得到结果");
        System.out.println(response);
        System.out.println("调用结束");        try {
            cyclicBarrier.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (BrokenBarrierException e) {
            e.printStackTrace();
        }
    }    public static void main(String[] args) {
        Demo5 demo5 = new Demo5();
        demo5.call();        try {
            demo5.cyclicBarrier.await();
        } catch (InterruptedException e) {
            e.printStackTrace();
        } catch (BrokenBarrierException e) {
            e.printStackTrace();
        }
        System.out.println("主线程内容");
    }
}
大家注意一下,CyclicBarrier和CountDownLatch仅仅只是类似,两者还是有一定区别的。比如,一个可以理解为做加法,等到加到这个数字后一起运行;一个则是减法,减到0继续运行。一个是可以重复计数的;另一个不可以等等等等。
另外,使用CyclicBarrier的时候要注意两点。第一点,初始化的时候,参数数字要设为2,因为异步调用这里是一个线程,而主线程是一个线程,两个线程都await的时候才能继续执行,这也是和CountDownLatch区别的部分。第二点,也是关于初始化参数的数值的,和这里的demo无关,在平时编程的时候,需要比较小心,如果这个数值设置得很大,比线程池中的线程数都大,那么就很容易引起死锁了。
Java异步调用转同步的5种方式的更多相关文章
- 5种必会的Java异步调用转同步的方法你会几种
		
转载请注明本文地址:https://www.jianshu.com/p/f00aa6f66281 源码地址:https://gitee.com/sunnymore/asyncToSync Sunny先 ...
 - 说说Java异步调用的几种方式
		
日常开发中,会经常遇到说,前台调服务,然后触发一个比较耗时的异步服务,且不用等异步任务的处理结果就对原服务进行返回.这里就涉及的Java异步调用的一个知识.下面本文尝试将Java异步调用的多种方式进行 ...
 - Java线程同步的四种方式详解(建议收藏)
		
 Java线程同步属于Java多线程与并发编程的核心点,需要重点掌握,下面我就来详解Java线程同步的4种主要的实现方式@mikechen 目录 什么是线程同步 线程同步的几种方式 1.使用sync ...
 - 实现web数据同步的四种方式
		
http://www.admin10000.com/document/6067.html 实现web数据同步的四种方式 1.nfs实现web数据共享 2.rsync +inotify实现web数据同步 ...
 - Java反射获取class对象的三种方式,反射创建对象的两种方式
		
Java反射获取class对象的三种方式,反射创建对象的两种方式 1.获取Class对象 在 Java API 中,提供了获取 Class 类对象的三种方法: 第一种,使用 Class.forName ...
 - 【Linux】多线程同步的四种方式
		
背景问题:在特定的应用场景下,多线程不进行同步会造成什么问题? 通过多线程模拟多窗口售票为例: #include <iostream> #include<pthread.h> ...
 - linux下实现web数据同步的四种方式(性能比较)
		
实现web数据同步的四种方式 ======================================= 1.nfs实现web数据共享2.rsync +inotify实现web数据同步3.rsyn ...
 - (转)SqlServer 数据库同步的两种方式 (发布、订阅),主从数据库之间的同步
		
最近在琢磨主从数据库之间的同步,公司正好也需要,在园子里找了一下,看到这篇博文比较详细,比较简单,本人亲自按步骤来过,现在分享给大家. 在这里要提醒大家的是(为了更好的理解,以下是本人自己理解,如有错 ...
 - SQL Server 2008 数据库同步的两种方式 (发布、订阅)
		
参考转载: SQL Server 2008 数据库同步的两种方式 (发布.订阅) 使用Sqlserver事务发布实现数据同步
 
随机推荐
- [转载]Java进程物理内存远大于Xmx的问题分析
			
进程物理内存远大于Xmx的问题分析 问题描述 最近经常被问到一个问题,”为什么我们系统进程占用的物理内存(Res/Rss)会远远大于设置的Xmx值”,比如Xmx设置1.7G,但是top看到的Res的值 ...
 - redis过期机制及排行榜
			
redis 内存数据集大小上升到一定大小的时候,就会施行数据淘汰策略.redis 提供 6种数据淘汰策略: volatile-lru:从已设置过期时间的数据集(server.db[i].expire ...
 - 前端知识体系:JavaScript基础-作用域和闭包-词法作用域和动态作用域
			
词法作用域和动态作用域 1.作用域: 作用域是指程序代码中定义变量的区域 JavaScript采用词法作用域,也就是静态作用域 2.词法作用域和动态作用域 因为JavaScript采用的是词法作用域, ...
 - partial 部分类
			
partial 关键字允许把类.结构.方法或接口放在多个文件中.一般情况下,一个类全部驻留在单个文件中.但有时,多个开发人员需要访问同一个类,或者某种类型的代码生成器生成了一个类的某部分,所以把类放在 ...
 - eclipse 设置字体高亮
			
在eclipse中不小心按错了什么键,使得变量的高亮显示没了. 其恢复方式如下: 选择:windows-> preferences->java->Editor->Mark Oc ...
 - python自动华 (四)
			
Python自动化 [第四篇]:Python基础-装饰器 生成器 迭代器 Json & pickle 目录: 装饰器 生成器 迭代器 Json & pickle 数据序列化 软件目录结 ...
 - [JSOI2018]潜入行动 (树形背包)
			
题目链接 题意: 外星人的母舰可以看成是一棵 n 个节点. n−1 条边的无向树,树上的节点用 1,2,⋯,n 编号.JYY 的特工已经装备了隐形模块,可以在外星人母舰中不受限制地活动,可以神不知鬼不 ...
 - 安装Discuz
			
1.下载Discuz 版本文件 http://download.comsenz.com/DiscuzX/3.2/Discuz_X3.2_SC_GBK.zip 2.下载PHP http://window ...
 - CF #589 (Div. 2)C. Primes and Multiplication 快速幂+质因数
			
题目链接:https://www.luogu.org/problem/CF1228C 问题可以转化为:求质数 $p$ 在 $1\sim n$ 中的每个数中的次幂之和. 因为 $p$ 是一个质数,只能由 ...
 - 【FTP】详解
			
FTP协议及工作原理 1. FTP协议 什么是FTP呢?FTP 是 TCP/IP 协议组中的协议之一,是英文File Transfer Protocol的缩写. 该协议是Internet文件传送的 ...