块设备是能随机存取装置固定大小的数据表设备。如硬盘;字符设备(如串口和键盘)它是按照字符流进入有序进行。不同之处在于是否足够的随机存取数据——这时候,你可以随心所欲地从一个位置跳到访问设备和位置。复杂多,实际上内核在块设备上下了大工夫——块I/O层。

基础概念

  • 块设备中最小的可寻址单元是扇区。
  • 文件系统的最小寻址单元是块。

  • 所谓的缓冲区是块在内存中的表示。
  • 对于一个缓冲区(块),内核须要知道它的控制信息。这时须要一个结构进行描写叙述——缓冲区头。

I/O调度机制

首先,这里的I/O调度指的是磁盘调度。磁盘寻址是非常耗时的一项操作。对于同一时候存在的各种磁盘I/O请求,我们须要利用一些I/O调度程序实现最省时的请求处理——I/O调度程序让整个请求队列按扇区增长方向有序排列。使全部请求按硬盘上扇区的排列顺序有序排列的目的不仅是为了缩短单独一次请求的寻址时间,更重要的优化在于,通过保持磁盘头以直线方向移动,缩短了全部请求的磁盘寻址时间。

该排序算法类似于电梯调度——电梯不能任意地从一层跳到还有一层,它应该向一个方向移动。当抵达了同一方向上的最后一层后,再掉头向还有一个方向移动。所以I/O调度程序称作电梯调度。

另外出了排序,合并也是一种降低磁盘寻址时间的方法。合并是将两个或多个请求结合成一个新请求,这样将多次请求的开销压缩成一次请求的开销,更重要的是,请求合并后仅仅须要传递给磁盘一条寻址命令。就能够訪问到请求合并前必须要多次寻址才干訪问完的磁盘区域,总之降低了磁盘寻址次数。

排序。降低寻址时间。而合并,降低寻址次数,二者共同作为I/O调度程序的准则。

以下看一下五个Linux实际使用的I/O调度程序。
(1)Linus电梯
当一个请求增加到队列中时。有可能发生四种操作:
  • 假设队列中已存在一个对相邻磁盘扇区操作的请求。那么新请求将和这个已经存在的请求合并成一个请求。
  • 假设队列中存在一个驻留时间过长的请求,那么新请求将被插入到队列尾部,以防止其它旧的请求饥饿发生。
  • 假设队列中以扇区方向为序存在合适的插入位置。那么新的请求将被插入到该位置,保证队列中的请求是以被訪问磁盘物理位置为序进行排列的。
  • 假设队列中不存在合适的请求插入位置。请求将被插入到队列尾部。
(2)终于期限I/O调度程序
对磁盘同一位置的连续请求造成较远位置的其它请求永远得不到执行机会,这是一种非常不公平的饥饿现象。

在最后期限I/O调度中。每一个请求都有一个超时时间,默认情况下。读请求的超时时间是500ms。写请求的是5s,最后期限I/O调度请求类似于Linus电梯,也以磁盘物理位置为次序维护请求队列,这个队列称为排序队列。当一个新请求递交给排序队列时,最后期限I/O调度程序在执行合并和插入请求时类似于Linus电梯,可是最后期限I/O同一时候也会以请求类型为根据将它们插入到额外队列中。

读请求按次序被插入到特定的读FIFO队列中,写请求按次序插入到特定的写FIFO队列中,尽管普通队列以磁盘扇区为序进行排列。可是这些队列是以FIFO(以时间为序)组织的,结果新队列总是被增加到队列尾部,对于普通操作来说。最后期限I/O将请求从排序队列头部取下。再推入到派发队列,由派发队列将请求提交给磁盘驱动。从而保证了最小化的请求寻址。

而最后期限I/O的超时体如今读写FIFO队列头的请求超时时。调度程序便会提取请求进行服务。这样不用操心有请求不能得到服务的不公平现象。

(3)预測I/O调度程序
基本上同终于期限。只是它能进行预測。假设读请求一个接着一个,对于终于期限来说要不停地为读请求寻址。而且中断正在进行的写请求,这会损害系统全局吞吐量。假设我们能等待,将一个接着一个的读请求进行统一寻址,这会避免大量的寻址操作。在一个请求提交后,预測会接着有I/O请求在等待期到来。从而有意等待片刻(不是一提交请求就立马返回处理其它请求),这便是预測I/O的巧妙之处。

这样的预測依靠一系列的启示和统计工作。

(4)全然公正的排队I/O调度程序
它给每个进程维护一个队列,请求增加队列后的操作同Linus电梯的四种操作,这事实上在进程级提供了公平。
(5)空操作的I/O调度程序
不进行排序。也不预測。它仅仅进行合并——将相邻请求合并。而且维护请求队列以近乎FIFO的顺序排列。

版权声明:本文博主原创文章。博客,未经同意不得转载。

最新文章

  1. 给Excel2013添加WebADI的Oracle加载项
  2. 使用原生JavaScript
  3. VB.Net中点击按钮(Button)会重复提交两次表单
  4. Linux下编译boost库和qt和ImageMagick
  5. js 去重 字符串 [123123,123123,345435,33467,45645,343467,879,45645]
  6. VC++中的DDX和DDV
  7. 小议common lisp程序开发流程 - Ever 17 - 博客频道 - CSDN.NET
  8. php下安装动态扩展库的相关事项
  9. no suitable driver found for jdbc:mysql//localhost:3306/..
  10. 给Ocelot做一个Docker 镜像
  11. 如何打开iPhone 中的heic格式照片
  12. Zipkin分布式跟踪系统介绍
  13. Docker(五)如何构建Dockerfile
  14. Android ImageView 的scaleType 属性图解
  15. js 对象数组查找元素常用方法
  16. 浅谈微服务架构、容器技术与K8S
  17. 由于想要实现下载的文件可以进行选择,而不是通过<a>标签写死下载文件的参数,所以一直想要使用JFinal结合ajax实现文件下载,但是ajax实现的文件下载并不能触发浏览器的下载文件弹出框,这里通过模拟表单提交实现同样的效果。
  18. 01-MySql的前戏
  19. 【转】通过blob获取图像并显示
  20. Jquery 网页转换为图片

热门文章

  1. ZJOI2002昂贵的聘礼题解
  2. 从零开始使用git第二篇:git的日常操作
  3. 2016最新CocoaPods安装与使用
  4. Java反射机制的简单应用
  5. go 保留小数若干位数
  6. ZOJ 3168 Sort ZOJ7 水
  7. Ubuntu10.04下安装Qt4和创建第一个Qt程序
  8. php计算两个坐标直线距离
  9. STS 3.6.4 SpringMVC 4.1.6 Hibernate 4.3.8 MySQL
  10. Android中动态设置GridView的列数、列宽和行高