实验5Linux文件操作之带缓存和非缓冲文件的读写.docx
《实验5Linux文件操作之带缓存和非缓冲文件的读写.docx》由会员分享,可在线阅读,更多相关《实验5Linux文件操作之带缓存和非缓冲文件的读写.docx(13页珍藏版)》请在冰豆网上搜索。
![实验5Linux文件操作之带缓存和非缓冲文件的读写.docx](https://file1.bdocx.com/fileroot1/2022-10/10/32bff8e5-c3e2-450f-a3a4-5cbd21f4041a/32bff8e5-c3e2-450f-a3a4-5cbd21f4041a1.gif)
实验5Linux文件操作之带缓存和非缓冲文件的读写
实验5Linux文件操作之带缓存和非缓冲文件的读写
学生姓名:
王祥真学号:
95专业班级:
计科143
实验类型:
□验证▥综合□设计□创新实验日期:
实验成绩:
一.实验目的
通过编写文件读写及上锁的程序,进一步熟悉Linux中文件I/O相关的应用开发,并且熟练掌握open()、read()、write()、fcntl()等函数的使用。
二.实验内容
1.用write,read,open等系统调用编写分别实现如下功能的程序(要求进行必要的出错检查):
(1)创建一个文件testfile.txt,文件内容从键盘输入;
(2)将testfile.txt的内容显示在屏幕上,并将testfile.txt的内容复制到一个新的文件file2.txt中。
实验代码:
#include
#include
#include
#include
#include
#include
#include
#include
intmain()
{
intfd,fdsrc,fddes,nbytes;
intnewret,n1,n2;
intflags=O_CREAT|O_TRUNC|O_WRONLY;
charsrc[50],des[30],buf[30];
charstr1[60];
charstr2[60];
strcat(str1,"cat");
printf("请输入要创建的源文件名字");
scanf("%s",src);
fd=creat(src,S_IRUSR|S_IWUSR);//创建源文件
n1=read(STDIN_FILENO,buf,80);
if(n1<0){
perror("readSTDIN_FILENO");
exit
(1);
}
fdsrc=open(src,flags,0644);//以读写方式打开
if(fdsrc<0){
exit
(1);
}
write(fdsrc,buf,n1);//写入
printf("输出刚创建源文件的内容\n");
strcat(str1,src);//catsrc;
newret=system(str1);//执行命令
close(fdsrc);//关闭指针
printf("请输入目标文件名:
");
scanf("%s",des);
fd=creat(des,S_IRUSR|S_IWUSR);
fdsrc=open(src,O_RDONLY);
if(fdsrc<0){
exit
(1);
}
fddes=open(des,flags,0744);
if(fddes<0){
exit
(1);
}
while((nbytes=read(fdsrc,buf,20))>0){
intz=write(fddes,buf,nbytes);
if(z<0){
perror("写文件出错");
}
}
close(fdsrc);
close(fddes);
printf("复制%s文件到%s文件成功!
\n",src,des);
return0;
}
实验结果:
2.在Linux中FIFO是一种进程之间的管道通信机制。
Linux支持完整的FIFO通信机制。
本实验内容,通过使用文件操作,仿真FIFO(先进先出)结构以及生产者-消费者运行模型。
本实验中需要打开两个虚拟终端,分别运行生产者程序(producer)和消费者程序(customer)。
此时两个进程同时对同一个文件进行读写操作。
因为这个文件是临界资源,所以可以使用文件锁机制来保证两个进程对文件的访问都是原子操作。
先启动生产者进程,它负责创建仿真FIFO结构的文件(其实是一个普通文件)并投入生产,就是按照给定的时间间隔,向FIFO文件写入自动生成的字符(在程序中用宏定义选择使用数字还是使用英文字符),生产周期以及要生产的资源数通过参数传递给进程(默认生产周期为1s,要生产的资源数为10个字符)。
后启动的消费者进程按照给定的数目进行消费,首先从文件中读取相应数目的字符并在屏幕上显示,然后从文件中删除刚才消费过的数据。
为了仿真FIFO结构,此时需要使用两次复制来实现文件内容的偏移。
每次消费的资源数通过参数传递给进程,默认值为10个字符。
三.实验环境
PC微机
Windows操作系统、虚拟机、Linux操作系统
四.实验步骤
1.编写代码实现要求的功能;
2.本实验问验证实验,但其中用到的文件操作函数和重要的文件锁操作:
1)实验流程图本实验的两个程序的流程图如图:
开始开始
(producer)(customer)
消费资源
创建FIFO结构文件(打印字符)
否
消费够了吗?
生产一个资源
是
上锁
上锁
将剩下的数据拷贝到
等待临时文件tmp中
一秒将“生产”的字符
写入到FIFO结构文件
用临时文件tmp覆盖
原数据文件,这样
模拟FIFO结构
解锁
解锁
生产完了吗?
删除临时文件
结束结束
图6.4节流程图
(2)代码
头部文件代码:
mylock.h
structmyflock
{
shortl_type;/*文件锁类型:
F_RDLOCK读取锁;F_WRLCK写入锁;F_UNLCK解锁*/
off_tl_start;/*相对位移量*/
shortl_whence;/*相对位移量的起点SEEK_SET;SEEK_CUR;SEEK_END:
*/
off_tl_len;/*加锁区域长度*/
pid_tl_pid;/**/
};
/*lock_set*/
intlock_set(intfd,inttype)
{
structmyflockold_lock,lock;
lock.l_whence=SEEK_SET;
lock.l_start=0;
lock.l_len=0;
lock.l_type=type;
lock.l_pid=-1;
/*判断文件是否可以上锁*/
fcntl(fd,F_GETLK,&lock);
if(lock.l_type!
=F_UNLCK)
{
/*判断文件不能上锁的原因*/
if(lock.l_type==F_RDLCK)/*该文件已有读取锁*/
{
printf("Readlockalreadysetby%d\n",lock.l_pid);
}
elseif(lock.l_type==F_WRLCK)/*该文件已有写入锁*/
{
printf("Writelockalreadysetby%d\n",lock.l_pid);
}
}
/*l_type可能已被F_GETLK修改过*/
lock.l_type=type;
/*根据不同的type值进行阻塞式上锁或解锁*/
if((fcntl(fd,F_SETLKW,&lock))<0)
{
printf("Lockfailed:
type=%d\n",lock.l_type);return1;
}
switch(lock.l_type)
{
caseF_RDLCK:
printf("Readlocksetby%d\n",getpid());
break;
caseF_WRLCK:
printf("Writelocksetby%d\n",getpid());
break;
caseF_UNLCK:
printf("Releaselockby%d\n",getpid());return1;
break;
default:
break;
}/*endofswitch*/
return0;
}
生产者程序的源代码:
producer.c
/*producer.c*/
#include
#include
#include
#include
#include
#include"mylock.h"
#defineMAXLEN10/*缓冲区大小最大值*/
#defineALPHABET1/*表示使用英文字符*/
#defineALPHABET_START'a'/*头一个字符,可以用'A'*/
#defineCOUNT_OF_ALPHABET26/*字母字符的个数*/
#defineDIGIT2/*表示使用数字字符*/
#defineDIGIT_START'0'/*头一个数字字符*/
#defineCOUNT_OF_DIGIT10/*数字字符的个数*/
#defineSIGN_TYPEALPHABET/*本实例用英文字符*/
constchar*fifo_file="./myfifo";/*!
"FIFO文件名*/
charbuff[MAXLEN];/*缓冲区*/
/*函数product()产生一个字符并写入仿真FIFO文件中*/
introduct(void)
{
intfd;
unsignedintsign_type,sign_start,sign_count,size;
staticunsignedintcounter=0;
/*打开!
"FIFO文件*/
if((fd=open(fifo_file,O_CREAT|O_RDWR|O_APPEND,0644))<0)
{
printf("Openfifofileerror\n");exit
(1);
}
sign_type=SIGN_TYPE;
switch(sign_type)
{
caseALPHABET:
/*英文字符*/
{
sign_start=ALPHABET_START;
sign_count=COUNT_OF_ALPHABET;
}
break;
caseDIGIT:
/*数字字符*/
{
sign_start=DIGIT_START;
sign_count=COUNT_OF_DIGIT;
}
break;
default:
{
return-1;
}
}/*endofswitch*/
sprintf(buff,"%c",(sign_start+counter));counter=(counter+1)%sign_count;
lock_set(fd,F_WRLCK