前言
上篇文章介绍了文件的物理结构并介绍了文件分配的三种方式——连续分配、链接分配和索引分配。
本文介绍操作系统对文件存储空间的管理。
本文内容
1 存储空间的划分与初始化
存储空间的划分:将物理磁盘划分为一个个文件卷(逻辑卷、逻辑盘)。
在存储空间初始化时,需要将各个文件卷划分为目录区、文件区。
(1) 目录区:主要存放文件目录信息(FCB)、用于磁盘存储空间管理的信息。
(2) 文件区主要存放文件数据。
有些系统支持超大型文件,可支持由多个物理磁盘组成一个文件卷。
2 存储空间管理——空闲表法
空闲表法:即用一张表记录磁盘中空闲的盘块。空闲表的表项由空闲盘的起始块号和空闲盘块数组成。如下图所示
如何分配磁盘块:与内存管理中的动态分区分配类似,为一个文件分配连续的存储空间。同样可以采用首次适应算法、最佳适应算法、最坏适应算法,临近适应算法来决定要为文件分配哪些区间。
空闲表法适用于连续分配方式。
例如,如果新创建的文件请求3个块,按照首次适用算法,从10号块开始有5个连续的块可以满足需求,所以把10、11、12三个块分配给文件,分配后的空闲盘块表如下
如何回收磁盘块:与内存管理中的动态分配很类似,当回收某个存储区时需要分为四种情况
(1) 回收区前后都没有相邻空闲区——为空闲盘块表增加一个表项即可。
(2 回收区前后都是空闲区——将回收区和回收区前后的两个空闲区合并为一个空闲区。
(3) 回收区前是空闲区——将回收区和前空闲合并为一个空闲区。
(4) 回收区后是空闲区——将回收区和后空闲合并为一个空闲区。
这里以回收区前后都是空闲区为例,磁盘是第一幅图的状态,如果回收21、22号磁盘块,那么回收后的空闲盘块表如下图所示。
其他情况与之类似,回收时注意表项能合并的话就需要合并。
3 存储空间管理——空闲链表法
空闲链表法分为两种:空闲盘块链和空闲盘区链
(1) 空闲盘块链:以盘块为单位组成一条空闲链。空闲盘块中存储着下一个空闲盘块的指针。
(2) 空闲盘区链:以盘区为单位组成一条空闲链。每个空闲盘区中的第一个盘区记录了盘区的长度、下一个盘区的指针。
下图分别表示空闲盘块链和空闲盘区链。
空闲盘区链中每个盘区中的第一个盘块记录了盘区长度和指向下一个盘区的指针。如21、22、23号3个盘块组成的盘区,21号盘块中记录了指向下一个盘区的指针(即指向6、7号盘块组成的盘区的指针)和盘区的长度3,其他的盘块类似。
3.1 空闲盘块链
操作系统保存着链头、链尾指针。
如何分配:如过某文件申请K个盘块,则从链头开始依次摘下K个盘块分配,并修改空闲链的链头指针。
如何回收:回收的盘块依次挂到链尾,并修改空闲链的链尾指针。
下图表示分配了3个盘块
下图表示回收了3个盘块
从上面可以看出,空闲盘块法适用于离散分配的物理结构。为文件分配多个盘块时可能要重复多次操作。
3.1 空闲盘区链
操作系统保存着链头、链尾指针。
如何分配:若某文件申请K个盘块,由于空闲盘区链将连续的盘块组成一个盘区,所以若某个盘区大小满足可以实现一次分配,同样可以采用首次适用、最佳适用等算法,从链头开始检索,按照一定的规则找到一个大小符合要求的空闲盘区分配给文件。若没有合适的连续空闲块,也可以将不同的盘区的盘同时分配给一个文件,同样分配后也需要修改相应的指针链和盘区大小等数据。
如何回收:若回收区和某个空闲盘区相邻,则需要将回收区合并到空闲盘区中。若回收区没有和任何空闲区相邻,将回收区作为一个单独的一个空闲盘区挂到链尾。同样也需要修改链表指针和盘区大小等信息。
下图表示按照首次适用算法分配3个盘区
回收过程和空闲盘块法的回收类似,这里不再赘述。
从上面可以看出,空闲盘区链对离散分配、连续分配都适用。为一个文件分配多个盘块时效率更高。
4 存储空间管理——位示图法
位示图:磁盘内存被划分为一个个磁盘块,可以用二进制位对应一个盘块。“0”代表盘块空闲,“1”代表盘块已分配。位示图一般用连续的“字”来表示,下图中一个字的字长是16位,字中的每一位对应一个盘块。因此可以用(字号,位号)对应一个盘块号。
注:字号和位号也可以从1开始,本例中都是从0开始的。
如何实现盘块号与(字号、位号)相互转换:
(字号,位号) = (i,j)的二进制位对应的盘块号 b = n*i + j.
b号盘块对应的字号 i = b / n,位号 j = b % n。
(1) (0,1)—> b = 16 * 0 + 1 = 1号块,从左侧图可以看出1号图已分配,同时位示图中(0,1)的二进制位也是1表示已分配。
(2) (1,10) —> b = 16 * 1 + 10 = 26号块,从左侧可以看出26号块是未分配的。
(3) b = 13 —> i = 13 / 16 = 0,j = 13 % 16 =13,对应的(字号,位号) = (0,13),从位示图可以查出对应的二进制位为0,表示未分配。
如何分配:若文件需要K个块,①顺序扫描位示图,找到K个相邻或不相邻的“0”;②根据字号、位号算出对应的盘块号,将相应的盘块分配给文件;③将相应的位设置为“1”。
如何回收:①根据回收的盘块号计算出对应的字号、位号;②将相应的二进制位设置为“0”。
从上面可以看出:位示图法对连续分配和离散分配都适用。
5 存储空间管理——成组链接法
空闲表法、空闲链表法不适用大型文件系统,因为空闲表或空闲联保可能过大。UNIX系统中采用了成组链接法对磁盘空闲块进行管理。这是将上述两种方法相结合的而形成的一种空闲管理方法。
文件卷的目录区中专门用一个磁盘块作为超级块,当系统启动时需要将超级块读入内存。并且要保证与外存中的“超过块”的数据一致。
(1) 超级块实际上是一个栈结构,用来存放当前可用的一组空闲盘块号(最多含100个号)以及栈中尚有空间的块号数量N。
(2) 文件区中的所有空闲盘块分为若干个组,假设所有空闲盘块号是201~9999(实际中盘块是不要求连续的,不要求连续的,不要求连续的,这里是为了表述方便,认为空闲盘块连续的),将这些盘块进行分组,每组100个。
(3) 每一组含有的盘块总数N和该组的所有盘号块记入前一组的第一个盘块中。这样各组的第一个盘块可以链成一条链。
(4) 如果是最后一组,那么将前一个分组的第一个盘块的中存放空闲盘块链的结束标志符,如“-1”表示下一组是最后一个分组了。
内存的分配过程:分配过程是从栈顶取出一空闲盘块号,将与之对应的盘块分配给用户,然后将栈顶指针下移一格,若该盘块号已是栈底(即第一个盘块),这是当前栈中最后一个可分配的盘块号。由于在该盘块号所对应的盘块中记有下一组可用的盘块号,因此,不能直接将它分配掉,需要将它记录的下一组信息保存下来,所以比须调用磁盘读过程,将栈底盘块号所对应盘块的内容读入栈中,作为新的盘块号栈的内容,并把原栈底对应的盘块分配出去(其中的有用数据已读入栈中)。然后,再分配一相应的缓冲区(作为该盘块的缓冲区)。最后,把栈中的空闲盘块数减1 并返回。
下面举例说明
如果此时新建一个文件需要一个磁盘块,那么此时第一组有100个空闲块,所以是足够分配的,将栈顶的盘块号即201号盘块对应的盘块分配出去,如下图
如果此时又创建一个新的文件,需要99个磁盘块,就需要将剩下的99个盘块全部分配出去,但是此时300号盘块记录了下一组信息,如果分配出去,信息就是丢失,所以需要将300号盘块从外存(磁盘)读入内存,将300号盘块记录的信息,写入空闲盘块号栈,然后才能将这99块空闲块分配出去。具体过程如下图所示
内存的回收过程:在系统回收空闲盘块时,须调用盘块回收过程进行回收。它是将回收盘块的盘块号记入空闲盘块号栈的顶部,并执行空闲盘块数加 1 操作。当栈中空闲盘块号数目已达 100 时,表示栈已满,便将现有栈中的100 个盘块号记入新回收的盘块中,再将其盘块号作为新栈底。
以分配的第一个图为例,201盘块被分配出去了,如果此刻有个文件被删除了,其占用的盘块是199号,系统需要回收这个盘块,发现此时空闲盘块号栈中记录空闲块数为99,直接将盘块号记录栈顶,将空闲盘块数加1即可。
如果此时又有一个文件被删除了,其占用的盘块是190,此时空闲盘块号数已经达到100了,就需要将现在空闲盘块栈中信息记入新回收的块中。
网友评论