在架构领域,我们经常听见别人说“异步”,另外大家在面试中也经常会遇到这方面的问题。
什么是异步?
说到.NET异步,先来说几个比较容易混淆的概念:
1、多线程:一般指同一进程中多个线程协作运行。在多核计算机中多个线程将有机会同时运行于多个核上,如果线程中进行的是计算,则行成并行计算。
2、并行:一般指并行计算,是说同一时刻有多条指令同时被执行,这些指令可能执行于同一CPU的多核上,或者多个CPU上,或者多个物理主机甚至多个网络中。
3、异步:与同步相对应,是指呼叫另一 *** 作后,不等待其结果,继续执行之后的 *** 作,若之后没有其他 *** 作,当前线程将进入睡眠状态,而CPU时间将有机会切至其他线程。在异步 *** 作完成后通过回调函数的方式获取通知与结果。
在Web中为什么要使用异步?
在web应用程序看到大量在启动时的并发请求或具有突发负载(其中会增加并发情况突然),进行异步web服务调用将增加你的应用程序的响应能力。异步请求采用相同量的时间来处理与同步请求。
例如,如果某个请求生成web服务调用,则需要两秒钟来完成,请求所执行的两秒内是否执行同步或异步。但是,在异步调用,线程则无需必须等待第一个请求完成时响应其他请求。因此,异步请求可以防止出现请求排队和线程池增长时有许多并发请求调用长时间运行的 *** 作。
异步的实现方式
异步的实现方式有多种,比如多线程。多线程将异步 *** 作放入另一线程中运行,通过轮询或回调方法得到完成通知。
多线程实现异步调用可以采用传统的Thread类来实现,Task来实现以及async/await关键字实现。
对于Thread来说,当多个异步方法需要协助时,代码将显得十分复杂,为此我们需要一个更好的异步模型,从.NET4开始,引入了新的异步模型。下面来看一段代码:
我们不再关心任务如何去开始,何时会结束,一切变成了一些有关或无关的任务。
对于封装一个由多个异步方法组合成的异步方法,组合异步方法调用后,按顺序调用第一个异步方法,紧接着,产生需要的结果Task后返回。异步方法完成时回调指定的方法,并按顺序继续调用,所有方法完成后,把运行的最终结果设置给结果Task,那么整个任务即完成。如果异步方法有返回值,那么组合的异步方法看上去会复杂一点。类似下图所示:
为了解决这个问题,微软提供了一个非常好用的语法糖:async/await关键字。下面我们来看一段代码:
他几乎和同步方法书写一致,代码量也比较少,程序的逻辑也很清晰,当然可读性更强。
总结
关于.NET网站异步处理分享到这里,这里做个总结:
1、.NET网站异步处理可以采用3中方式实现:Thread,Task,async/await;
2、Task出现是为了Thread的不足,async/await出现也是为了弥补Task的不足。大家有没有发现这个关系就像一条线一样。其实说到底这些都是多线程技术,底层都是Thread实现,只不过微软为了代码的可读性以及逻辑的清晰性,做了语法糖,从而更加方便开发人员使用。
3、在C#5中,添加了async/await关键字,使得上面遗留的问题得以解决,而且重点是,用起来非常简单!
4、NETFramework4引入了异步编程概念,ASP.NET4.5支持任务,于是就有了async/await。
以上就是我的观点,对于这个问题大家是怎么看待的呢?欢迎在下方评论区交流~我是科技领域创作者,十年互联网从业经验,欢迎关注我了解更多科技知识!在JAVA平台 实现异步调用的角色有如下三个角色:调用者 提货单 真实数据一个调用者在调用耗时 *** 作 不能立即返回数据时 先返回一个提货单 然后在过一断时间后凭提货单来获取真正的数据 去蛋糕店买蛋糕 不需要等蛋糕做出来(假设现做要很长时间) 只需要领个提货单就可以了(去干别的事情) 等到蛋糕做好了 再拿提货单取蛋糕就可以了 public class Main { public static void main(String[] args) {
System out println( main BEGIN )
Host host = new Host()
Data data = host request( A )
Data data = host request( B )
Data data = host request( C )
System out println( main otherJob BEGIN )
try {
Thread sleep( )
} catch (InterruptedException e) {
}
System out println( main otherJob END )
System out println( data = + data getContent())
System out println( data = + data getContent())
System out println( data = + data getContent())
System out println( main END )
}
}
这里的main类就相当于 顾客 host就相当于 蛋糕店 顾客向 蛋糕店 定蛋糕就相当于 发请求request 返回的数据data是FutureData的实例 就相当于提货单 而不是真正的 蛋糕 在过一段时间后(sleep一段时间后) 调用data getContent() 也就是拿提货单获取执行结果
下面来看一下 顾客定蛋糕后 蛋糕店做了什么
public class Host {
public Data request(final int count final char c) {
System out println( request( + count + + c + ) BEGIN )
// ( ) 建立FutureData的实体
final FutureData future = new FutureData()
// ( ) 为了建立RealData的实体 启动新的线程
new Thread() {
public void run() {
//在匿名内部类中使用count future c
RealData realdata = new RealData(count c)
future setRealData(realdata)
}
} start()
System out println( request( + count + + c + ) END )
// ( ) 取回FutureData实体 作为传回值
return future
}
}
host( 蛋糕店 )在接到请求后 先生成了 提货单 FutureData的实例future 然后命令 蛋糕师傅 RealData去做蛋糕 realdata相当于起个线程去做蛋糕了 然后host返回给顾客的仅仅是 提货单 future 而不是蛋糕 当蛋糕做好后 蛋糕师傅才能给对应的 提货单 蛋糕 也就是future setRealData(realdata)
下面来看看蛋糕师傅是怎么做蛋糕的
建立一个字符串 包含count个c字符 为了表现出犯法需要花费一些时间 使用了sleep
public class RealData implements Data { private final String content
public RealData(int count char c) {
System out println( making RealData( + count + + c + ) BEGIN )
char[] buffer = new char[count]
for (int i = i <counti++) {
buffer[i] = c
try {
Thread sleep( )
} catch (InterruptedException e) {
}
}
System out println( making RealData( + count + + c + ) END )
ntent = new String(buffer)
}
public String getContent() {
return content
}
}
现在来看看 提货单 future是怎么与蛋糕 content 对应的:
public class FutureData implements Data { private RealData realdata = null
private boolean ready = false
public synchronized void setRealData(RealData realdata) {
if (ready) {
return// 防止setRealData被调用两次以上
}
this realdata = realdata
this ready = true
notifyAll()
}
public synchronized String getContent() {
while (!ready) {
try {
wait()
} catch (InterruptedException e) {
}
}
return realdata getContent()
}
}
顾客做完自己的事情后 会拿着自己的 提货单 来取蛋糕
System out println( data = + data getContent())
这时候如果蛋糕没做好 就只好等了
while (!ready) { try {
wait()
} catch (InterruptedException e) {
}
//等做好后才能取到
return realdata getContent()
程序分析
对于每个请求 host都会生成一个线程 这个线程负责生成顾客需要的 蛋糕 在等待一段时间以后 如果蛋糕还没有做好 顾客还必须等待 直到 蛋糕被做好 也就是
future setRealData(realdata)执行以后 顾客才能拿走蛋糕
每个线程只是专门负责制作特定顾客所需要的 蛋糕 也就是顾客A对应着蛋糕师傅A 顾客B对应着蛋糕师傅B 即使顾客B的蛋糕被先做好了 顾客A也只能等待蛋糕师傅A把蛋糕做好 换句话说 顾客之间没有竞争关系
lishixinzhi/Article/program/Java/gj/201311/27425软件模块之间的调用关系可以分为两大类:即同步调用和异步调用。在同步调用中,一段代码(主调方)调用另一段代码(被调方),主调方必须等待这段代码执行完成返回结果后,才能继续往下执行,所以,同步调用是一种阻塞式调用,主调方代码一直阻塞等待直到被调方返回为止。同步调用相对比较直观,也是大部分编程语言直接支持的一种调用方式。但是,同步调用在处理比较耗时的情况下会严重影响程序性能,影响人机交互的瞬时反应。例如,某个程序需要访问数据库获取大量数据,然后根据这些数据进行一系列处理,将处理结果显示在程序主窗口。由于数据库访问和大量数据的处理都是耗时的工作,在这个工作完成之前,处理结果迟迟不能显示,用户点击鼠标也不会立即得到响应,让用户感到整个程序显得很沉重。面对这样一些需要比较长时间才能完成的应用场景,我们需要采用一种非阻塞式调用方式,即异步调用方式。在异步调用中,主调方调用被调方后,不等待对方返回结果就继续执行后续代码,被调方执行完毕后,通过某种手段通知调用方:结果已经出来,请酌情处理。我们可以对上面的例子改用异步调用将问题轻松化解:把整个耗时的工作放进一个单独的线程,由主调方启动此线程后继续执行后续代码,线程在背后悄悄地处理费时的工作,当工作完成,采用回调的方式通知主调方工作完成,主调方将结果显示在主窗口。经过这样的处理,主界面继续进行自己的工作而不必死等,就不会造成界面响应迟钝。
在实现异步调用机制时,除了线程之外,还要用到回调。回调是一种双向调用,也就是,被调方在被调用时也会调用主调方的代码。在异步调用中,被调方需要在工作完成时通知主调方,即调用主调方的接口,这一机制通过回调实现。回调和异步调用的关系非常紧密,回调是异步调用的基础[1]。
本文理论联系实际,首先阐述如何使用Java实现回调机制,然后进一步阐述使用Java回调和线程实现异步调用,最后,阐述在异步调用中如何处理超时问题。
1 Java回调机制的实现方法
实现Java回调,需要做如下三件事情:
(1)定义一个回调接口CallbackInterface
接口中声明回调方法handle,如图1所示,回调方法就是一个普通的方法,接收一个消息字符串或者一个封装了数据的事件。
(2)定义一个类实现回调接口
这个类其实就是消息接收者和处理者,也就是调用方,回调方法是消息发生时实际处理消息的方法,此处简化为一条打印语句。
(3)定义消息通知者
消息通知者也就是被调用方必须具备两种能力,第一,它必须知道谁是消息接收者,第二,当消息发生时,它能够回调这些接收者的回调方法。为了获得这两种能力,消息通知者首先必须提供一个注册方法register, 通过注册的方式来注册多个对此消息或事件感兴趣的对象。然后提供一个消息通知方法notifyMessage,在这个方法中调用所有消息接收者的回调方法。具体代码如图3所示。
比如用一个可变数组List用于保存消息接收者,注册的过程实际上是将消息接收者添加到这个数组,以备在需要通知消息的时候调用这些消息接收者的回调方法。
使用Java回调和线程实现异步调用
线程是一个独立的执行流,其本质是程序中一段并发执行的代码。在异步调用机制中引入线程,在线程中完成耗时的工作,其目的是让调用方的主线程继续执行后续代码而不需要等待被调方的结果返回。由于不需要等待,这样我们就等于同时做了两件事情,而这两件事情分别是在不同的执行流中执行,主调者在当前的主线程中执行,被调者在另外一个线程中执行,因此提高了程序的效率,避免了界面的响应迟钝。当被调者执行完成后,仍然采用回调通知主调者。
例如 LongTimeWorker是一个用于完成耗时工作的线程,同时又是消息通知者。其耗时工作在run方法中完成,另外提供一个注册方法register, 和一个消息通知方法notifyMessage,在run方法的最后,即耗时工作完成以后,调用notifyMessage将消息广播出去。
3 异步调用中超时问题的处理
异步调用通常都要加入超时机制,因为我们总是希望在一个指定的时间范围内返回一个结果,即使没有得到结果也该有个超时通知。这时我们需要使用“限时线程回调方式”,它在原有线程回调的基础上加上一个计时器Timer以计算消耗的时间,如果时间期限到了任务还没有执行完成即中断线程,并将超时消息广播出去。LongTimeWorker类需要修改部分的代码如图8和图9所示。
首先LongTimeWorker线程类增加了一个构造方法,其参数是超时时间timeout,构造方法的主要任务是创建一个定时器,每秒钟计时一次,若超时时间到则终止本线程,并广播超时消息。LongTimeWorker线程类的第二个改变发生在其run方法中,线程一启动立即开始计时,完成工作后停止计时,并广播消息。
4 结束语
异步调用是一种非阻塞式调用方式,用于在处理比较耗时的任务时保证程序性能不受到影响。实现异步调用的关键在于要解决三个技术难题,它们分别是程序阻塞问题、异步消息的传递问题和超时问题。本文介绍的方法采用并发线程、回调机制和计时器使上述问题得到了圆满解决。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)