资源描述
高级操作系统实验报告
姓 名:
学 号:
专 业:
任 课 教 师:
2014年01 月18
目 录
一、课程设计题目和目的 1
二、课程设计要求 1
三、程序设计思想 1
四、文件系统的实现 2
1.数据结构设计 2
2.程序功能图 3
3.实体关系图 3
4.数据流图 4
5.程序流程图 5
(1) .建立文件:create(文件名,记录长度) 6
(2) .写文件:write(文件名,开始位置,字符串) 7
(3) .读文件:read(文件名,开始位置,长度) 8
(4) .显示文件所有内容 type(文件名) 8
(5) .删除文件delete(文件名) 9
(6) .重命名文件ren(文件名,新文件名) 10
(7) .查询文件属性ask(文件名) 11
(8) .关闭文件close(文件名) 12
五、程序运行结果及分析 13
六、课程设计总结 15
七、参考文献 16
八、附录 17
一、课程设计题目和目的
通过模拟文件系统的实现,深入理解操作系统中文件系统的理论知识, 加深对教材中的重要算法的理解。同时通过编程实现这些算法,更好地掌握操作系统的原理及实现方法,提高综合运用各专业课知识的能力。
二、课程设计要求
通过组长分工,我主要完成了如下几个功能的操作:
1.创建文件:从命令中得到文件名,得到该文件的文件长度,建立文件。修改目录表。
4.读文件:read [文件名] [显示开始字节] [显示的字节数] ,直接显示所需要的字节数。
5.写文件:write [文件名] [插入的位置] [插入的内容]
6.修改属性:修改文件属性 文件名,文件权限,文件长度等。
三、程序设计思想
阅读操作系统方面的书籍,了解操作系统的文件系统原理。结合分析课程设计要求,确定实体以及它们之间的关系。实体关系有三张表(磁盘空间分配表、文件表、打开文件表)、一个模拟磁盘的数组、命令服务和用户构成。用户负责输入命令。命令服务实现命令的解释、命令检查、命令帮助以及调用相关模块执行相应的命令功能。
文件夹打开是则把文件夹名称及其地址压入打开文件夹栈,文件关闭则把文件夹名称及其地址从打开文件夹栈中抛出。
文件打开则把文件的名称及其父指针写到文件列表同时置文件打开标志为1,文件关闭则把文件从打开列表中删除,同时置文件打开指针为0,文件读取和写入都要检查文件是否在文件打开列表中,未打开文件不能读写,只读文件不能写,只写文件不能读。
文件夹和文件创建,文件夹和文件的创建首先检验目录是否为空,为空则把文件夹或文件连接到该目录下,不为空则把检查目录下是否有同名文件夹或文件,有则提示创建不成功,没有则把文件夹或文件连接到该目录下的最后一个子节点,作为它的兄弟节点。
文件夹和文件的删除,文件夹下没有打开的文件或文件没有打开才能删除,否则删除失败,删除文件夹时利用了中序历遍来删除子树。
四、文件系统的实现
1.数据结构设计
通过分析课程设计要求,具体设计出如下数据结构:
① typedef struct uof //文件属性
{
char filename[14]; //文件名字
int mode; //文件的权限0-readonly;1-writeonly;2-read/write
int length; //文件长度
int addr; //物理块号
int state; //0-建立,1-建立
int readptr;
};
通过结构体,将文件名字、文件在磁盘的开始位置、文件长度、文件最大长度、文件类型、创建时间结合在一起。文件类型,本模拟程序使用txt类型。设置一个线性表来存储文件。
2.程序功能图
模拟文件系统
创建文件
删除文件
条件读取
写入文件
查询属性
显示内容
显示目录
重命名文件
关闭文件
文件系统提供的文件操作有建立文件(create)、删除文件(delete)、条件读取文件(read)、写入文件(write)、查询文件的属性(ask)、显示文件所有内容(type)、重命名文件(ren)、关闭文件(close)。可以通过键盘输入命令来模拟文件的操作。通过exit命令退出程序。
3.实体关系图
用 户
命令服务
模拟磁盘
磁盘空间分配表
文件表
打开文件表
命令服务使得用户能够输入命令,在需要时提供命令的帮助。同时能够分析命令,调用相应的命令模块对模拟磁盘、磁盘空间分配表、文件表、打开文件表进行操作。磁盘空间分配表记录模拟磁盘的使用情况。文件表记录文件的信息和在磁盘里的位置等信息。打开文件表记录已打开的文件,对应文件表中的文件信息,和文件表里的文件节点类似,记录了文件在模拟磁盘中的信息。
4.程序流程图
模拟文件系统提供的文件操作有建立(create),读取(read),显示(type),删除(detele),写入(write),关闭(close),重命名(ren)和查询(ask)。在模拟程序中可从键盘上输入文件操作命令来模拟各用户程序中所调用的各种文件操作,用一个结束命令(exit)停止程序的执行。
开始
系统格式化
Y/N
/
输入命令
命令表中有该命令?
分析命令
显示:命令帮助
无
创建
删除
读取
显示
重命名
写入
查询
列表
关闭
退出程序(exit)
执行命令
执行命令
(1) .创建文件:create(文件名,记录长度)
模拟文件系统进行“创建文件”的处理流程如下:
开始
查询文件表
文件表中有名字?
返回
有
显示重命
申请磁盘空间
空间申请成功?
显示失败
否
返回
无
在磁盘分配表中登记
登记:长度=0,最大长度=申请长度,开始位置=申请节点位置
空闲情况=占用
在文件表中登记
登记:长度=0,最大长度=建立长度,开始位置=分配位置
文件类型=txt,时间=当前时间
在打开文件表中登记
登记:信息和文件表中类似
显示创建成功
返回
(2) .写文件:write(文件名,开始位置,字符串)
模拟文件系统进行“建立文件”的处理流程如下:
开始
查询打开文件表
查询文件表
在打开文件表里?
不在
在文件表里?
显示无文件
返回
不在
写进模拟磁盘
在
在
读取文件记录
write参数合法?
合法
显示成功
返回
显示参数非法
非法
(3) .读文件:read(文件名,开始位置,长度)
模拟文件系统进行“读取文件”的处理流程如下:
开始
查询文件表
在打开文件表里?
不在
在文件表里?
显示无文件
返回
不在
在
在
读取文件记录
read参数合法?
非法
显示参数非法
返回
根据参数读取模拟磁盘里的记录
返回
合法
五.实验程序说明:
1.创建文件:
2.打开文件:
3.写文件:
4.修改文件属性:
六、课程设计总结
通过本次的课程设计,使我能够正确运用操作系统课程中所学的基本理论和知识,加深了对文件系统基本概念的理解,以及磁盘文件系统的文件操作。还有让我感受挺深的是对软件工程方法的应用。设计一个软件,先要做好需求分析,这一点很重要,如果没有分析好需求,到软件设计的最后,发现所做的功能不符合要求,那么一切都得重做,前面所有的努力都付诸东流。在程序设计的开始,由于分析工作做得不够深入和细致,吃了点小苦头。对于这样一个小设计来说,都会吃苦头,要是大工程更是无法想像,有可能会项目失败。以后得加强对软件工程的学习。另外在运用C语言的时候,感觉有点生疏,在组织语言时时而出错,在编程和调试的过程中,经常会出现意想不到的问题,并非每个问题都可以从相关资料中找到解决方法,有些问题是无法预料到的,这就需要通过自己理性的分析得出问题的解决方案。
在设计过程中,查询了不少相关资料,不断的发现问题、提出问题、解决问题。在对自己所编写的源程序段的纠错的过程中,使我更好的理解了操作系统中文件系统的理论知识,同时在编程时用到了模块化的设计思想,这种编程方法可以使我们的编程变的更简单,可以使我们的查错与纠错变的更方便。总的来说通过这次的设计的学习使我学到了很多在平时的学习中学不到的很多东西,通过这次课程设计,使我对操作系统和编程产生兴趣,我想我会在这条路上继续前进下去。我相信,只要不断的严格要求自己,注意培养自己的思维能力,就一定会有更大更辉煌的发展和提高。
七、参考文献
[1][美]Abraham Silberschatz Peter Baer Galvin Greg Gagne 郑扣根 译.OPERATING SYSTEM CONCEPTS[Sixth Edition] 操作系统概念.高等教育出版社,2004,01
[2](美)Roger S.Pressman 著 梅宏 译.软件工程-实践者的研究方法.机械工业出版社,2002,09
[3]蔡启先.C语言程序设计教程(第二版).重庆大学出版社,2003,07
[4]朱战立.数据结构.西安电子科技大学出版社,2003,05
八、附录
1.程序源代码及注释
////////////////////////////////////////////////////////////
// 编译工具:Visual c++ 6.0 //
// //
// 作者:~小楚*smill~ 、蓝色的天空 //
// 邮件:qvb20974151@ //
// //
// 复制、传播请保留作者信息,谢谢! //
// //
////////////////////////////////////////////////////////////
#include <stdio.h>
#include <conio.h>
#include <string.h>
#include <stdlib.h>
#include <time.h>
void fileCreate(char *fileName,int fileLenght); //创建文件
void fileDel(char *fileName); //删除文件
void fileRead(char *fileName,int postion,int lenght);
void fileWrite(char *fileName,int position,char *s); //函数声明
void fileAsk(char *fileName); //查询文件属性
void fileType(char *fileName); //显示文件内容
void fileRen(char *fileName,char *newName); //重命名文件
void fileDir(); //列出所有文件
void fileClose(char *fileName); //关闭一个文件
int requestDisk(int *startPosition,int *maxLength); //分配磁盘空间
/////////////////////////////////
//模拟磁盘空间
char disk[10000];
///////////////////////////////
//磁盘空间管理
struct freeDiskTable
{
int start; //开始位置
int length; //占用长度
int maxLength; //最大长度
int useFlag; //使用标志,1为占用,0为空闲
};
//////////////////////////////////
//文件表
struct fileTable
{
char fileName[20]; //文件名字
int start; //开始位置
int length; //文件长度
int maxLength; //最大长度
char fileKind[8]; //文件种类,此默认为txt
struct tm *timeinfo; //文件创建时间
};
////////////////////////////////
//打开文件表
struct openFileTable
{
char fileName[20]; //文件名字
char fileKind[8]; //文件类型
int start; //文件开始位置
int length; //文件长度
int maxLength; //最大长度
int openCount; //打开文件的进程数
struct tm *timeinfo; //文件创建时间
};
//////////////////////////////
typedef struct freeDiskTable LinDataType; //定义链表的数据类型
#include "LinList.h" //链表操作集合
SLNode *freeDiskTableHead; //定义磁盘分配表链表 头指针
#define MaxSize 100 //定义线性表 最大长度
typedef struct fileTable SeqDataType; //定义线性表数据类型
#include "SeqList.h" //线性表操作集合
SeqList L; //文件表,模拟文件操作嘛,文件数不多,故可以用 线性表,简化操作^_^
struct openFileTable OFT[10]; //打开文件表数组
int OFT_count=0; //打开文件表占用长度
void main()
{
struct orderTable //命令表
{
char name[8];
};
char orderInput[30]; //存储用户输入的命令字符串
char orderName[8]; //命令名字
char fileName[20]; //命令参数1,是文件名字
char parameter2[6]; //命令参数2,是个数字
char parameter3[30]; //命令参数3,是字符串或数字
struct orderTable u[9]; //命令表
int i=0;
int flag=0; //用户输入串 读写位置 标志
int flagOrder=-1;
/////////////////////////////////////
LinListInitiate(&freeDiskTableHead);//磁盘空间分配键表初始化
struct freeDiskTable temp;
temp.length=0;
temp.maxLength=10000;
temp.start=0;
temp.useFlag=0;
if(!LinListInsert(freeDiskTableHead,0,temp))
{
printf("初始磁盘空闲表失败!");
system("pause");
exit(0);
}
/////////////////////////////////////
//文件表初始化
SeqListInitiate(&L); //初始化 文件表
/////////////////////////////////////
for(i=0;i<2;i++)
for(int j=0;j<2;j++)
{
system("color fc");
printf("\n\n\n\n\n\n\n\n\n 计 046 何珠举 操作系统课程设计\n");
for(int k=0;k<30000000;k++);
system("cls");
system("color f3");
printf("\n\n\n\n\n\n\n\n\n 计 046 何珠举 操作系统课程设计\n");
for(k=0;k<30000000;k++);
system("cls");
}
system("color 17"); //设置背景色 和 前景色
////////////////////////////////////////////////////////////
//初始化命令表
strcpy(u[0].name,"create");
strcpy(u[1].name,"delete");
strcpy(u[2].name,"read");
strcpy(u[3].name,"write");
strcpy(u[4].name,"ask");
strcpy(u[5].name,"type");
strcpy(u[6].name,"ren");
strcpy(u[7].name,"dir");
strcpy(u[8].name,"close");
////////////////////////////////////////////////////////////
while(1)
{
printf("\n");
printf("|******************************************************|\n");
printf("|* create-创建 delete-删除 read-读取 write-写入 *|\n");
printf("|* ask-查询 type-显示 ren-重命名 dir-显示文件 *|\n");
printf("|* close-关闭文件 *|\n");
printf("|* 输入'命令 ?'获得帮助,例如'create ?',exit-退出程序 *|\n");
printf("|******************************************************|\n");
printf("请输入命令:>");
fflush(stdin); //清空缓冲区
gets(orderInput); //获取用户输入串
if(!strcmp(orderInput,"exit"))exit(0); //当用户输入 exit 退出程序
system("cls"); //清屏
for(i=0;orderInput[i]!=' '&&orderInput[i]!='\0';i++)//以空格为界 分解用户输入串
{
orderName[i]=orderInput[i];
}
flag=i;//保存读写指针
if(flag>6) //命令名长度 小于6
{
printf("输入的命令有误,请重新输入!\n");
continue ;
}
orderName[i]='\0';
flagOrder=-1; //初始化标志位,flagOrder 指明命令在命令表中的位置
for(i=0;i<9;i++)
{
if(!strcmp(orderName,u[i].name))
{
flagOrder=i; //记录命令的位置,以便访问
break;
}
}
if(flagOrder==-1)
{
printf("你输入的命令有误!请重新输入!\n");
continue ;
}
/////////////////////////////////////////////////////
//以空格为界,分解字符串,分解出第一个字符串放到fileName中
for(i=flag+1;orderInput[i]!=' '&&orderInput[i]!='\0';i++)
{
fileName[i-flag-1]=orderInput[i];
}
fileName[i-flag-1]='\0'; //为字符串数组 置一个结束标志
/////////////////////////////////////////////////////
flag=i; //下一个字符串 读写指针
if(!strcmp(fileName,"?")) //显示命令帮助
{
switch(flagOrder)
{
case 0:
printf("创建一个文件 格式 create [filename][filelength]\n");
printf("例如: create a1 1000 ,将创建名为a1,长度为1000字节的文件\n");
break;
case 1:
printf("删除一个文件 格式 delete [filename] \n");
printf("例如: delete a1,将删除名为a1的文件\n");
break;
case 2:
printf("读取一个文件 格式 read [filename][postion][length]\n");
printf("例如: read a1 8 3,从a1文件第8个字节开始,显示3字节内容\n");
break;
case 3:
printf("写一个文件 格式 write [fileName][postion][conten] \n");
printf("例始: write a1 15 test,从a1文件第15字节开始写入 test 字符串\n");
break;
case 4:
printf("显示文件的属性,格式 ask [filename]\n");
printf("例如 ask a1 ,将显示文件 a1的属性\n");
break;
case 5:
printf("显示文件的所有内容,格式 type [filename]\n");
printf("例如 type a1,将显示文件 a1的所有内容\n");
break;
case 6:
printf("重命名一个文件,格式 ren [oldFileName][newFileName]\n");
printf("例如ren a1 b1 ,将a1改名为b1\n");
break;
case 7:
printf("显示所有文件,例如 dir,将显示所有文件\n");
break;
case 8:
printf("关闭文件,格式 close [fileName]\n");
printf("例如,close a1,将关闭文件a1\n");
break;
default:
printf("命令错误");
}
getch();
}
else
{
switch(flagOrder)//对相应的命令 进行参数处理
{
case 0:
{
int Temp[4]={1,10,100,1000};
int sum=0;
int len=0;
for(i=flag+1;orderInput[i]!=' '&&orderInput[i]!='\0';i++)
{
parameter2[i-flag-1]=orderInput[i];
}
parameter2[i-flag-1]='\0'; //为字符串数组 置一个结束标志
flag=i;
len=strlen(parameter2); //求字符串数组的长度
for(i=0;i<len;i++) //把第二个参数 字符串变成整数
sum=sum+((int)parameter2[len-i-1]-48)*Temp[i];
fileCreate(fileName,sum); //将命令和参数 传给建文件模块执行
}
break;
case 1:
fileDel(fileName); //删除文件
break;
case 2:
{
int Temp[4]={1,10,100,1000};
int sum=0;
int len=0;
int sum2=0;
//////////////////////////////////////////////////////////
for(i=flag+1;orderInput[i]!=' '&&orderInput[i]!='\0';i++)
{
parameter2[i-flag-1]=orderInput[i];
}
parameter2[i-flag-1]='\0';
flag=i;
len=strlen(parameter2); //求字符串数组的长度
for(i=0;i<len;i++) //把第二个参数 字符串变成整数
sum=sum+((int)parameter2[len-i-1]-48)*Temp[i];
///////////////////////////////////////////////////////////
for(i=flag+1;orderInput[i]!=' '&&orderInput[i]!='\0';i++)
{
parameter3[i-flag-1]=orderInput[i]; //把第三个参数变成整型
}
parameter3[i-flag-1]='\0';
flag=i;
len=strlen(parameter3);
for(i=0;i<len;i++)
sum2=sum2+((int)parameter3[len-i-1]-48)*Temp[i];
fileRead(fileName,sum,sum2); //调用读文件模块
}
break;
case 3:
{
int Temp[4]={1,10,100,1000};
int sum=0;
int len=0;
//////////////////////////////////////////////////////////
for(i=flag+1;orderInput[i]!=' '&&orderInput[i]!='\0';i++)
{
parameter2[i-flag-1]=orderInput[i];
}
parameter2[i-flag-1]='\0';
flag=i;
len=strlen(parameter2); //求字符串数组的长度
for(i=0;i<len;i++) //把第二个参数 字符串变成整数
sum=sum+((int)parameter2[len-i-1]-48)*Temp[i];
///////////////////////////////////////////////////////////
for(i=flag+1;orderInput[i]!=' '&&orderInput[i]!='\0';i++)
{
parameter3[i-flag-1]=orderInput[i]; //获取第三个参数,为字符串
}
parameter3[i-flag-1]='\0';
flag=i;
//////////////////////////////////////////////////////////
fileWrite(fileName,sum,parameter3); //调用写文件模块
}
break;
case 4:
fileAsk(fileName); //查询文件属性
break;
case 5:
fileType(fileName); //显示文件内容
break;
case 6:
{
for(i=flag+1;orderInput[i]!=' '&&orderInput[i]!='\0';i++)
{
parameter2[i-flag-1]=orderInput[i];
}
parameter2[i-flag-1]='\0';
flag=i;
fileRen(fileName,parameter2);
}
break;
case 7:
fileDir();
break;
case 8:
fileClose(fileName);
break;
default:
printf("错误!没有可执行的命令");
}
}
}
}
//以上主函数,主函数进行数据结构的定义 分析用户输入的命令。
//提供命令帮助。把命令和参数,传递给执行模块
//////////////////////////////////////////////////////////////////////
void fileCreate(char *fileName,int fileLength)//没有写到打开文件表
{
int i=0;
time_t rawtime;
int startPosition=0; //文件开始位置,等磁盘分配函数 返回值
int maxLength=fileLength;
struct fileTable temp;
for(i=0;i<L.size;i++)
if(!strcmp(fileName,L.list[i].fileName)) //判断是否重名,如果重名直接返回
{
printf("文件重名,请取别的名字!\n");
return ; //文件重名,退出创建函数
}
if(!requestDisk(&startPosition,&maxLength))
{
printf("申请磁盘空间失败!无法建立文件!\n");
return ; //申请磁盘空间失败,退出创建函数
}
strcpy(temp.fileName,fileName);
strcpy(temp.fileKind,"txt"); //设置文件类型为 txt
temp.length=0; //创建的时候,还没写入内容,是空文件,长度设为0
temp.maxLength=fileLength; //文件的最大长度
temp.start=startPosition; //文件的开始位置
time(&rawtime); //获取时间
temp.timeinfo=localtime(&rawtime); //把时间写到结构体里去
if(!SeqListInsert(&L,L.size,temp)) //把文件 插入到文件表
{
printf("插入文件表失败!\n");
system("pause");
exit(0); //把文件插入到 文件表,如果失败,退出程序
}
printf("成功创建文件!\n\n");
printf("===========================================\n");
printf("文件名 长度 最大长度 类型 开始位置\n");//显示刚建立的文件
printf("%s ",temp.f
展开阅读全文