1、 实验5 Linux文件操作之带缓存和非缓冲文件的读写学生姓名:王祥真 学号:6103114095 专业班级: 计科143 实验类型: 验证 综合 设计 创新 实验日期: 2017.4.20 实验成绩: 一实验目的通过编写文件读写及上锁的程序,进一步熟悉 Linux 中文件 I/O 相关的应用开发, 并且熟练掌握 open()、read()、write()、fcntl()等函数的使用。二实验内容1. 用write, read, open等系统调用编写分别实现如下功能的程序(要求进行必要的出错检查):(1)创建一个文件testfile.txt,文件内容从键盘输入;(2)将testfile.txt
2、的内容显示在屏幕上,并将testfile.txt的内容复制到一个新的文件file2.txt中。实验代码:#include#include#include#include#include#include#include#includeint main() int fd, fdsrc,fddes,nbytes; int newret ,n1,n2; int flags=O_CREAT | O_TRUNC | O_WRONLY; char src50,des30,buf30; char str160; char str260; strcat(str1,cat ); printf(请输入要创建的源文件
3、名字); scanf(%s,src); fd=creat(src,S_IRUSR|S_IWUSR); /创建源文件 n1=read(STDIN_FILENO,buf,80); if(n10) perror(read STDIN_FILENO); exit(1); fdsrc=open(src,flags,0644); /以读写方式打开 if(fdsrc0) exit(1); write(fdsrc,buf,n1);/写入 printf(输出刚创建源文件的内容n); strcat(str1,src);/cat src; newret=system(str1);/执行命令 close(fdsrc)
4、;/ 关闭指针 printf(请输入目标文件名: ); scanf(%s,des); fd=creat(des,S_IRUSR|S_IWUSR); fdsrc=open(src,O_RDONLY); if(fdsrc0) exit(1); fddes=open(des,flags,0744); if(fddes0) int z=write(fddes,buf,nbytes); if(z0) perror(写文件出错); close(fdsrc); close(fddes); printf(复制%s文件到%s文件成功!n,src,des); return 0;实验结果:2. 在 Linux 中
5、FIFO 是一种进程之间的管道通信机制。Linux 支持完整的 FIFO通信 机制。 本实验内容,通过使用文件操作,仿真 FIFO(先进先出)结构以及生产者-消费者运行模型。 本实验中需要打开两个虚拟终端,分别运行生产者程序(producer)和消费者程序(customer)。此时两个进程同时对同一个文件进行读写操作。因为这个文件是临界资源,所以可以使用文件锁机制来保证两个进程对文件的访问都是原子操作。 先启动生产者进程,它负责创建仿真 FIFO 结构的文件(其实是一个普通文件)并投入生产,就是按照给定的时间间隔,向 FIFO 文件写入自动生成的字符(在程序中用宏定义选择使用数字还是使用英文字
6、符),生产周期以及要生产的资源数通过参数传递给进程(默认生产周期为 1s,要生产的资源数为 10 个字符)。 后启动的消费者进程按照给定的数目进行消费,首先从文件中读取相应数目的字符并在屏幕上显示,然后从文件中删除刚才消费过的数据。为了仿真 FIFO 结构,此时需要使用两次复制来实现文件内容的偏移。每次消费的资源数通过参数传递给进程,默认值为10 个字符。 三实验环境PC微机Windows 操作系统、虚拟机、Linux操作系统四实验步骤 1. 编写代码实现要求的功能;2. 本实验问验证实验,但其中用到的文件操作函数和重要的文件锁操作:1)实验流程图 本实验的两个程序的流程图如图: 开始开始(p
7、roducer)(customer)消费资源 创建FIFO结构文件(打印字符)否消费够了吗?生产一个资源是上锁上锁将剩下的数据拷贝到等待临时文件tmp中一秒将“生产”的字符写入到FIFO结构文件用临时文件tmp覆盖原数据文件,这样模拟FIFO结构解锁解锁 生产完了吗? 删除临时文件结束结束图 6.4 节流程图(2)代码 头部文件代码: mylock.hstruct myflock short l_type; /*文件锁类型: F_RDLOCK 读取锁;F_WRLCK 写入锁;F_UNLCK 解锁 */ off_t l_start; /*相对位移量*/ short l_whence; /*相对位
8、移量的起点SEEK_SET;SEEK_CUR; SEEK_END: */ off_t l_len; /* 加锁区域长度 */ pid_t l_pid; /* */ ;/* lock_set */ int lock_set(int fd, int type) struct myflock old_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);
9、 if (lock.l_type != F_UNLCK) /* 判断文件不能上锁的原因 */ if (lock.l_type = F_RDLCK) /* 该文件已有读取锁 */ printf(Read lock already set by %dn, lock.l_pid); else if (lock.l_type = F_WRLCK) /* 该文件已有写入锁 */ printf(Write lock already set by %dn, lock.l_pid); /* l_type 可能已被 F_GETLK 修改过 */ lock.l_type = type; /* 根据不同的 type
10、 值进行阻塞式上锁或解锁 */ if (fcntl(fd, F_SETLKW, &lock) 0) printf(Lock failed:type = %dn, lock.l_type); return 1; switch(lock.l_type) case F_RDLCK: printf(Read lock set by %dn, getpid(); break; case F_WRLCK: printf(Write lock set by %dn, getpid(); break; case F_UNLCK: printf(Release lock by %dn, getpid(); re
11、turn 1; break; default: break; /* end of switch*/ return 0;生产者程序的源代码: producer.c/* producer.c */ #include #include #include #include #include #include mylock.h #define MAXLEN 10 /* 缓冲区大小最大值*/#define ALPHABET1 /* 表示使用英文字符 */#define ALPHABET_START a /* 头一个字符,可以用 A*/ #define COUNT_OF_ALPHABET 26 /* 字母字
12、符的个数 */#define DIGIT 2 /* 表示使用数字字符 */#define DIGIT_START 0 /* 头一个数字字符 */#define COUNT_OF_DIGIT 10 /* 数字字符的个数 */#define SIGN_TYPE ALPHABET/* 本实例用英文字符 */const char *fifo_file = ./myfifo;/* ! FIFO 文件名 */char buffMAXLEN;/* 缓冲区 */* 函数product() 产生一个字符并写入仿真 FIFO 文件中 */ int roduct(void) int fd; unsigned int
13、 sign_type, sign_start, sign_count, size; static unsigned int counter = 0; /* 打开! FIFO 文件 */ if (fd = open(fifo_file, O_CREAT|O_RDWR|O_APPEND, 0644) 0) printf(Open fifo file errorn); exit(1); sign_type = SIGN_TYPE; switch(sign_type) case ALPHABET: /* 英文字符 */ sign_start = ALPHABET_START; sign_count =
14、 COUNT_OF_ALPHABET; break; case DIGIT: /* 数字字符 */ sign_start = DIGIT_START; sign_count = COUNT_OF_DIGIT; break; default: return -1; /*end of switch*/ sprintf(buff, %c, (sign_start + counter); counter = (counter + 1) % sign_count; lock_set(fd, F_WRLCK); /* 上写锁*/ if (size = write(fd, buff, strlen(buff
15、) 1) /* 第一个参数表示生产周期 */ sscanf(argv1, %d, &time_step); if (argc 2) /* 第二个参数表示需要生产的资源数 */ sscanf(argv2, %d, &time_life); while (time_life-) if (product() 0) break; sleep(time_step); exit(EXIT_SUCCESS);消费者程序的源代码: customer.c /* customer.c */ #include #include #include #include #include #include mylock.h
16、 #define MAX_FILE_SIZE100 * 1024 * 1024 /* 100M*/const char *fifo_file = ./myfifo; /* 仿真 FIFO 文件名 */ const char *tmp_file = ./tmp; /* 临时文件名 */ /* 资源消费函数customing */int customing(const char *myfifo, int need) int fd; char buff; int counter = 0; if (fd = open(myfifo, O_RDONLY) 0) printf(Function custo
17、ming errorn);return -1; printf(Enjoy:); lseek(fd, SEEK_SET, 0); while (counter need) while (read(fd, &buff, 1) = 1) & (counter need) fputc(buff, stdout); /* -.就是在屏幕上/0的显示 */ counter+; fputs(n, stdout); close(fd); return 0;/* myfilecopy()函数: 实现从 sour_file 文件的 offset 偏移处开始将 count 个字节数据复制到 dest_file 文件
18、 */ int myfilecopy(const char *sour_file,const char *dest_file, int offset, int count, int copy_mode) int in_file, out_file; int counter = 0; char buff_unit; if (in_file = open(sour_file, O_RDONLY|O_NONBLOCK) 0) printf(Function myfilecopy error in source filen); return -1; if (out_file = open(dest_f
19、ile, O_CREAT|O_RDWR|O_TRUNC|O_NONBLOCK, 0644) 0) printf(Function myfilecopy error in destination file:); return -1; lseek(in_file, offset, SEEK_SET); while (read(in_file, &buff_unit, 1) = 1) & (counter count) write(out_file, &buff_unit, 1); counter+; close(in_file); close(out_file); return 0; /* cus
20、tom()函数:实现 FIFO 消费者 */ int custom(int need) int fd; /* 对资源进行消费,need 表示该消费的资源数目 */ customing(fifo_file, need); if (fd = open(fifo_file, O_RDWR) 1) /* 第一个参数指定需要消费的资源数目,默认值为 10 */ sscanf(argv1, %d, &customer_capacity); if (customer_capacity 0) custom(customer_capacity); exit(EXIT_SUCCESS);(3) 分别编译生产者程序
21、producer.c 和消费者程序customer.c(4) 确保编译没有错误后,先在控制台终端1上运行生产者程序:./producer 1 20再在控制台终端2上运行消费者程序:./customer 5观察两终端的输入输出情况实验结果:循环出现,所以只截图了其中一小部分。五、实验报告和要求 按照实验步骤要求完成本实验后,对实验结果截图,完成实验报告,给出实验总结。 通过本次实验,我了解了Linux环境下文件的基本IO操作。首先要使用open函数得到一个文件描述符fd,可以给open函数指定一些文件打开的参数,比如权限、模式等等。然后对于文件的所有操作其实都是针对文件描述符的操作,使用read和write来进行读写操作。系统默认有三个文件描述符:0(标准输入)、1(标准输出)、2(标准输出),可以看做是普通文件,来对其进行读写操作。11