聊聊同步和异步(2)

传递回调函数


1、使用Javascript编写
function complete(information){
        console.log(information);
}
function servlet(command){
        console.log("调用业务组件")
        service(command,complete);
}
function service(command,callBack){
        setTimeout(function(){
            console.log(command);
            callBack("业务组件完成调用");
        },1000);
}
//用户调用业务处理
servlet("select * from user_table where username = wangbinghua");
javascript_async_flow.png

Servlet() 作为用户调用的方法,其通知业务组件service(),并不知道业务组件何时调用完毕,因此将complete()回调函数作为参数,调用业务组件。等待业务组件处理完毕业务之后,再次调用complete()函数,表明已经完成业务调用。

2、使用Java编写
java_async.png

在java中无法传递函数,因此将接口作为参数进行传递,从而达到传递函数的目的。

interface CallBack{
        //回调函数
        public void callBack(String result);
}

用户主线程,调用业务组件。而业务主线程驱动ServletProcess类的invokeService方法,在调用业务组件的同时,开启一条线程来处理业务逻辑。因主线程驱动的ServletProcess类无法得知异步线程何时才能完成业务逻辑处理。所以,将回调函数所在的接口作为参数传递给业务逻辑所处的异步线程。在异步线程完成之后,再次调用callBack方法,表明业务逻辑完成处理。

class ServletProcess implements CallBack{

        private ServiceProcess serviceProcess;
        public ServletProcess(ServiceProcess serviceProcess){
            this.serviceProcess = serviceProcess;
        }

        public void dealOtherRequest(){
            System.out.println("接受其他用户的请求");
        }
        public void invokeService(final String information){

            System.out.println("用户线程开始:" + new Date());

            //开启异步线程调用业务处理组件(耗时)
            new Thread(new Runnable() {
                public void run() {

                    System.out.println("异步线程开始");
                    //调用业务组件
                    serviceProcess.dealService(information,ServletProcess.this);

                }
            }).start();

            //接受其他用户的请求
            this.dealOtherRequest();
            System.out.println("用户线程结束:" + new Date());
        }

        //业务组件完成后,调用该方法
        public void callBack(String result) {
            System.out.println(result);
        }
}

异步线程驱动的业务组件:

class ServiceProcess{

        public void dealService(String information,ServletProcess servletProcess){

            //处理业务
            try {
                Thread.sleep(3000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            System.out.println("处理业务:" + information);
            System.out.println("异步线程结束");

            //处理完毕之后,通知ServletProcess组件
            servletProcess.callBack("处理数据,渲染页面");
        }
}
3、类比Servlet异步处理
线程池.png

如果使用同步处理,那么用户每次请求一次,就需要从线程池中获取一个线程进行处理用户的请求。那么在同步的条件下,都是由这一个线程同时进行请求处理和业务处理。如果业务处理比较耗时,那么线程就会进行阻塞。此时,有更多的用户进行请求,线程池中的线程在极端情况下全部阻塞,那么就无法处理用户的请求。用户必须等待之前的业务处理的完成,很大程度上影响系统的吞吐量。因此提倡采用servlet的异步处理。

servlet通知完耗时业务组件处理业务之后,马上返回到线程池中,而不进行等待。后续的操作由回调函数或者事件监听器完成。这样,接下来更多的用户请求,就会充分利用线程池中的线程。

servlet_async.png

AsyncServlet异步调用业务组件处理业务逻辑,则其通知AsyncTask异步线程调用业务组件,然后立即返回。与此同时,Web容器线程将AsyncContext对象传递给AsyncTask异步线程。当异步线程处理业务完毕之后,将调用AsyncContext对象的complete方法或者dispach方法,表明业务处理完毕。

@WebServlet(name = "Servlet",urlPatterns = {"/us"},asyncSupported = true)
public class AsyncServlet extends HttpServlet {

    protected void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        AsyncContext asyncContext = request.startAsync();
        asyncContext.start(new AsyncTask(asyncContext));
    }
}

class AsyncTask implements Runnable{

    private AsyncContext asyncContext;
    public AsyncTask(AsyncContext asyncContext){
        this.asyncContext = asyncContext;
    }

    public void run() {
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("deal some things!");
        this.asyncContext.dispatch("/async.jsp");
//        this.asyncContext.complete();
    }
}

3、类比WebSocket异步处理
websocket_async.png

**
WebSocket的java服务器端要向客户端发送消息,可能发送这个消息非常耗时,那么此时会造成服务器端程序阻塞,使得服务器端的处理性能急剧下降。因此,可以对消息的发送进行异步处理。即WebSocket对应的Java API中的Async对象向服务器端发送消息时,调用send方法,其只是通知send方法,立即返回。异步线程(使用Future接口)来负责向客户端发送消息,此时容器主线程并不知道什么时候异步线程可以发送消息完毕。因此,在使用异步线程调用send方法的同时,将SendHandler接口传递给异步线程。当异步线程发送消息完毕时,则调用SendHandler接口的onResult方法,表明异步线程已经发送消息完毕,从而让容器主线程感知到。
**

 @OnMessage
  public void receiveMessage(Session session,String message,@PathParam("loginName") String loginName)  
          throws IOException {

        System.out.println("服务器收到的信息为:" + message);

        session.getAsyncRemote().sendText(SendInformationAsync.sendInfo(), new SendHandler() {
          //服务器向客户端发送数据完毕之后,则调用SendHandler接口的onResult方法
            public void onResult(SendResult result) {
                if(result.isOK()){
                    System.out.println("信息发送完毕");
                }
            }
        });
 }

使用监听器


1、使用Javascript编写
javascript_listener.png

用户主线程调用servlet方法,而servlet方法调用业务组件service。此时用注册一个事件的监听器,即事件发生之后,调用callBack方法。用户在servlet方法中调用service方法,立即返回,并不知道service方法中的业务何时处理完成。利用事件监听器,在service方法中的业务处理完成之后,出发刚才注册的事件,即可调用callBack方法。

function servlet(command){
        //调用业务组件
        console.log("调用业务组件");
        service(command);
}

function callBack(){
        console.log("渲染页面");
}

 function service(command){
        //业务组件
        setTimeout(function(){
            //处理业务
            console.log("开始处理业务");
            console.log(command);
            console.log("处理业务完毕")
           //触发事件
           $("#event").trigger("click");
       },1000);
}

$("#event").on("click",callBack);
servlet("select * from user_table where username = wangbinghua");
2、类比servlet异步处理
servlet_listener.png

在Web容器主线程中,调用业务组件,注册一个异步线程的监听器。该监听器主要监听四个事件,success,timeout,error,startAsync。Web容器的主线程调用业务组件,则开启一个异步线程,立即返回。主线程并不知道异步线程是否已经完成了业务处理。因此,异步线程在完成了业务处理之后,AsyncContext对象调用complete或者dispatch方法,即触发了success事件。触发该事件之后,立即调用监听器的onSuccess方法,表明异步线程已经完成业务处理。

@WebServlet(name = "Servlet",urlPatterns = {"/us"},asyncSupported = true)
public class AsyncServlet extends HttpServlet {
    protected void doPost(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

    }

    protected void doGet(HttpServletRequest request, HttpServletResponse response)
            throws ServletException, IOException {

        response.setCharacterEncoding("utf-8");

        final AsyncContext asyncContext = request.startAsync();

        //注册事件监听器
        asyncContext.addListener(new AsyncListener() {

            //异步线程业务处理完成之后,调用该方法
            public void onComplete(AsyncEvent asyncEvent) throws IOException {
                try {
                    asyncContext.getRequest().getRequestDispatcher("/async.jsp").
                            forward(asyncContext.getRequest(),asyncContext.getResponse());
                } catch (ServletException e) {
                    e.printStackTrace();
                }
                System.out.println("异步线程完成");
            }

            public void onTimeout(AsyncEvent asyncEvent) throws IOException {
                System.out.println("onTimeout");
            }

            public void onError(AsyncEvent asyncEvent) throws IOException {
                System.out.println("onError");
            }

            public void onStartAsync(AsyncEvent asyncEvent) throws IOException {
                System.out.println("onStartAsync");
            }
        });
        asyncContext.start(new AsyncTask(asyncContext));

    }
}

class AsyncTask implements Runnable{

    private AsyncContext asyncContext;
    public AsyncTask(AsyncContext asyncContext){
        this.asyncContext = asyncContext;
    }

    public void run() {
        try {
            Thread.sleep(3000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        System.out.println("deal some things!");
//        this.asyncContext.dispatch("/async.jsp");
        this.asyncContext.complete();
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,561评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,218评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,162评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,470评论 1 283
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,550评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,806评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,951评论 3 407
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,712评论 0 266
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,166评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,510评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,643评论 1 340
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,306评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,930评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,745评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,983评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,351评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,509评论 2 348

推荐阅读更多精彩内容

  • 从三月份找实习到现在,面了一些公司,挂了不少,但最终还是拿到小米、百度、阿里、京东、新浪、CVTE、乐视家的研发岗...
    时芥蓝阅读 42,212评论 11 349
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,633评论 18 139
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,598评论 18 399
  • 这世界从来都不缺乏创意,缺乏的只是将创意付诸实际的勇气每一部被称作概念机的设计,都倾注了消费者对生产厂商的期许,那...
    24e2f6668318阅读 236评论 0 0
  • 有人问 为什么我遇不到能够坚定的爱我的人 有时候只是他刚好需要 我也刚好在 这世间的爱情哪有那么多一往而深 深爱总...
    未曾饶过岁月阅读 171评论 0 0