《实验4-文件操作.docx》由会员分享,可在线阅读,更多相关《实验4-文件操作.docx(15页珍藏版)》请在第壹文秘上搜索。
1、实验4、文件操作学生姓名:李亚军学号:6100412196专业班级:卓越计科121班1 .实验目的通过编写文件读写及上锁的程序,进一步熟悉1.inux中文件I/O相关的应用开发,并且熟练掌握open。、read。、write。、fbntl()等函数的使用。2 .实验内容在1.inux中HFO是一种进程之间的管道通信机制。1.imIX支持完整的FlFo通信机制。本实验内容,通过使用文件操作,仿真FIFO(先进先出)结构以及生产者-消费者运行模型。本实验中需要打开两个虚拟终端,分别运行生产者程序(PrOdUCer)和消费者程序(CUStOmer)。此时两个进程同时对同一个文件进行读写操作。因为这个
2、文件是临界资源,所以可以使用文件锁机制来保证两个进程对文件的访问都是原子操作。先启动生产者进程,它负责创建仿真FIFO结构的文件(其实是一个普通文件)并投入生产,就是按照给定的时间间隔,向FIFO文件写入自动生成的字符(在程序中用宏定义选择使用数字还是使用英文字符),生产周期以及要生产的资源数通过参数传递给进程(默认生产周期为1s,要生产的资源数为10个字符)。后启动的消费者进程按照给定的数目进行消费,首先从文件中读取相应数目的字符并在屏幕上显示,然后从文件中删除刚才消费过的数据。为了仿真FIFO结构,此时需要使用两次复制来实现文件内容的偏移。每次消费的资源数通过参数传递给进程,默认值为IO个
3、字符。3 .实验步骤(1)实验流程图本实验的两个程序的流程图如图:上箱将利下的敬检拷贝到等待椎时文件m中一秒将“生产”的字符写入到FlFo结构文件用临时文件tnas原我摄文件,这律模粒FIFOtiW*排KW生产完了喝?热鞋临时文件笈克结东图6.4节流程图(2)代码头部文件代码:mylock.hstructmyflock(shortHype;*文件锁类型:F-RD1.OCK读取锁;F.WR1.CK写入锁;F_UN1.CK解锁*/off_tl_s(art;/*相对位移量*/short!,whence;芦相对位移量的起点SEEK_SET;SEEK_CUR;SEEK.END:*/OfCtlJen;/*
4、加锁区域长度*/pid_tl_pid;*/;*lock_set*/intlock_set(intfd,inttype)(structmyflockoldOCk,lock;lock.l-whence=SEEK-SET;lock.l-start=0;lock.lIen0;lock.Hype=type;lock.l_pid=-1;*判断文件是否可以上锁*/fcntl(fd.F-GET1.K.&lock);if(IockJjype!=F_UN1.CK)(*判断文件不能上锁的原因*/if(lock.l_type=F_RD1.CK)产该文件已有读取锁*/primf(Readlockalreadysetby
5、%dnlock.l_pid);elseif(lock.ljype=F.WR1.CK)产该文件已有写入锁*/(printf(Writelockalreadysetby%dn,lock.l_pid);)*1.type可能已被F_GET1.K修改过*/lock.l_type=type;*根据不同的type值进行阻塞式上锁或解锁*/if(fcntl(fU,F.SET1.KW,&lock)O)(printf(1.ockfailed:type=%dn,lock.Hype);return1;)Switch(Iock-1.lype)caseF_RD1.CK:printf(Readlocksetby%dn,ge
6、tpid();break;caseF.WR1.CK:printf(Writelocksetby%dn,getpidO);break;caseF-UN1.CK:printf(Releaselockby%dn,getpid();return1;break;default:break;*endofswitch*/return0;)生产者程序的源代码:producers*producers*/#include#include#include#include#include#includemylock-h/MefineMAX1.ENIO#defineA1.PHABETI#defineA1.PHABET_
7、START,a,#defineCOUNT_OF_A1.PHABET*缓冲区大小最大值号*表示使用英文字符*/*头一个字符,可以用A*/26/*字母字符的个数*/SdefineDIGIT2#defineDIGIT_STARTV#defineCOUNT_OF_DIGIT#defineSIGN_TYPEnstchar*fifo_fileCharbufllMAX1.EN;10A1.PHABET=ymyfifo;/*/*缓冲区/f*表示使用数字字符*/*头一个数字字符*/六数字字符的个数*/*本实例用英文字符*/!FIFO文件名*/*函数PnXlUCt()产生一个字符并写入仿真FIFO文件中*/into
8、duct(void)intfd;unsignedintSignjype,sign-start,sign-ut,size;staticunsignedintcounter=0;*打开!”FlFo文件/if(fd=OPen(IC1.CREATlo_RDwRlO_APPEND,0644)O)(PrintiVOpenfitbfilee11rn);exit(l);ISignJyPe=SIGN_TYPE;switch(sign_type)CaseA1.PHABET:*英文字符*/(SigI1.Start=A1.PHABET_START:sign_coun(=8UNTjDFA1.PHABET;break;c
9、aseDIGIT:*数字字符*/Sign_start=DIG-START;sign_count=COUNT_OF_DIGIT;)break;default:return-1;*endofswitch*/Sprintftbuff;%c,(sign_start+counter);counter=(counter+1)%sign_count;lock,set(ftlF_WR1.CK);*上写锁*/if(size=write(fij,buflf,strlen(buff)1)/*第一个参数表示生产周期*/sscanf(argvl,%d,&iime_s(ep);if(argc2)/*第二个参数表示需要生产
10、的资源数*/SSCan出arp2,%d”,&time_life);while(timejife-)(if(product()0)break;SleeP(IimJSteP);exil(EX,SUCCESS);消费者程序的源代码:Customenc*Customerx/include#include#include#include#include#include,mylock.h#defineMAX_H1.E_SIZE100*1024*1024*100M*/constchar*fifb-file=,.myfifo;*仿真FIFO文件名*/constchar*tmp_file=.tmp;*临时文件名
11、*/*资源消费函数CUstoming*/intcustoming(constchar*myfifb,inineed)intfd;charbuff;intcounter=0;if(fd=open(myfifo,O-RDON1.Y)0)(PrintfCFUnCtioncustomingerrorn);return-1;)printf(Enjoy:);lseek(fd,SEEK_SET,0);while(counterneed)(while(read(fid,&buff,1)=1)&(counterneed)fputc(buff,stdout);*-.就是在屏幕上/0的显示*/counter+;fp
12、uts(nstdout);close(fd);return0;*myfilecopyO函数:实现从sourJile文件的OffSet偏移处开始将count个字节数据复制到dest_file文件*fintmyfilecopy(constChar*sour_file,constchar*dest-file,intoffset,intcount,intcopy-mode)intin_file,out_file;intcounter=0;charbuff_unit;if(inJile=open(sour.file,O_RDON1.Y|O_NONB1.OCK)0)(PrintfrFUnCtiOnmyfi
13、lecopyerrorinsourcefilenr);return-I:)if(out-file=open(dest-file,0_CREAT10_RDWR|0_TRUNC|0_N0NB1.0CK.0644)0)(Prinlf(Functionmyfilecopyerrorindestinationfile:);return-I;lseek(in_file.offset,SEEK_SET);while(read(in-file,&buff_unit.1)=1)&(countercount)WriIe(OU1.file,&bulT_unit.1);counter+;close(in_file);close(out-file);return0;*custom。函数:实现FIFO消费者*/intcustom(in(need)(intfd;*对资源进行消费,need表示该消费的资源数目*/customing(fifb-file.need);if(ftl=open(fifb-file,O.RDWR)0)(PrinIffFunctionmyfilecopyerrorinsource_file: