同步/异步、阻塞/非租塞是两个不用的概念,并不是说一提到同步就是阻塞,一提到异步就是非租塞。同步与阻塞、或者异步与非租塞没有直接联系
同步/异步关注的是消息通知时机,阻塞/非租塞关注的线程等待调用结果时的状态(参考:https://www.zhihu.com/question/19732473 Yi Lu的回答)
1 概念
1.1 同步/异步
同步就是需要调用端主动去检查是否获取到了消息(举例体现在代码里就是errorcode = recvfrom(xxxxx),errorcode就是同步等待返回的消息),对于异步,调用端不用主动去检查是否有消息,被调用端会主动向调用端发送消息,或者调用回调函数
1.2 阻塞/非阻塞
阻塞直接提现就是代码执行到某个某句就卡住不动,不往下执行了,非阻塞刚好相反
2 分类
可以分为四类:同步阻塞、同步非租塞、异步阻塞、异步非租塞
2.1 同步阻塞
执行投送消息代码阻塞、等待获取返回消息,效率最低,但理解起来比较简单
代码体现:
errorcode = recvfrom(xxxx),errorcode是等待返回结果,是同步的体现,执行recvfrom时代码卡住,是阻塞的体现
举例说明:
小明一直盯着下载进度条,到 100% 的时候就完成
同步体现在:等待下载完成通知;
阻塞体现在:等待下载完成通知过程中,不能做其他任务处理;
2.2 同步非阻塞
举例说明:
小明提交下载任务后就去干别的,每过一段时间就去瞄一眼进度条,看到 100% 就完成
同步体现在:等待下载完成通知;
非阻塞体现在:等待下载完成通知过程中,去干别的任务了,只是时不时会瞄一眼进度条(小明必须要在两个任务间切换,关注下载进度)
代码体现:(用recvfrom函数来说)
调用recvfrom函数后如果发现数据没有准备好,则立即返回,这时可以执行一些其他的命令,但是需要不断的调用recvfrom函数去检查数据是否准备好,若准备好了,则阻塞执行该段代码(这里的数据准备好是指从网络缓存接受数据、将数据从网络缓存拷贝到系统内核、将数据从系统内核拷贝到用户内存这三个过程)
具有代表性意义的就是IO非阻塞模型
2.3 异步阻塞
异步阻塞是可以被阻塞住的,只不过它不是在投送消息时阻塞,而是在等待返回消息通知时阻塞,比如select 函数,假如传入的最后一个timeout参数为NULL,那么如果所关注的时间没有一个被处罚,程序就会一直阻塞在这个select函数调用处
举例说明:
异步阻塞:小明换了个有下载完成通知功能的软件,下载完成就“叮”一声。不过小明仍然一直等待“叮”的声音(看起来很傻,不是吗)
异步体现在:下载完成“叮”一声通知;
阻塞体现在:等待下载完成“叮”一声通知过程中,不能做其他任务处理
个人感觉异步阻塞没有什么意义
2.4 异步非阻塞
效率最高
举例说明:
仍然是那个会“叮”一声的下载软件,小明提交下载任务后就去干别的,听到“叮”的一声就知道完成了
异步体现在:下载完成“叮”一声通知;
非阻塞体现在:等待下载完成“叮”一声通知过程中,去干别的任务了,只需要接收“叮”声通知即可(软件处理下载任务,小明处理其他任务,不需关注进度,只需接收软件“叮”声通知,即可)
也就是说,同步/异步是“下载完成消息”通知的方式(机制),而阻塞/非阻塞则是在等待“下载完成消息”通知过程中的状态(能不能干其他任务)