1 #include <unistd.h> 2 #include <stdio.h> 3 #include <sys/mman.h> 4 #include <fcntl.h> 5 #include <stdlib.h> 6 7 typedef struct//定义了一个包含下面两个数据类型的结构体 8 { 9 int integer; 10 char string[24]; 11 } RECORD;//别名为RECORD 12 13 #define NRECORDS (100)//宏定义NRECORDS为(100) 14 15 int main()//主函数 16 { 17 RECORD record, * mapped;//分别实现了一个record结构体,和一个指向结构体的指针mapped 18 int i,f; 19 FILE * fp;//定义了流指针fp 20 21 fp=fopen("records.dat","w+");//以修改的方式打开文件,并将文件长度截断为0,没有的话就创建 22 for (i=0;i<NRECORDS; i++)//进入一个条件循环,循环100次 23 { 24 record.integer=i;//给结构体的证书部分赋值为i 25 sprintf(record.string, "RECORD-%d",i);//把格式化的输出传送到指定的地方,-代表左对齐 26 fwrite(&record, sizeof(record), 1, fp);//把赋值完成的record写入fp中,写入一个指定长度 27 } 28 fclose(fp);//关闭打开的流指针 29 30 31 32 fp=fopen("records.dat","r+");//以写的方式打开文件,并创建文件流fd 33 fseek(fp,43*sizeof(record),SEEK_SET);//设置文件流的下一次读写的位置,最后一个参数表示绝对位置 34 fread(&record,sizeof(record), 1, fp);//从fd中读取一个指定长度,到record结构体中 35 36 record.integer=143; //设置结构体中第一个元素的值为143 37 sprintf(record.string,"RECORD-%d",record.integer);//同样的把这个值按照左对齐方式传递给结构体的第二个元素 38 39 fseek(fp,43*sizeof(record),SEEK_SET);//重新设置文件流的下一次读写的位置 40 fwrite(&record,sizeof(record),1,fp);//给文件流写入一个指定长度的record内容 41 fclose(fp);//关闭打开的文件流 42 43 44 45 f=open("record.dat",O_RDWR);//使用系统调用open,以读写的方式打开文件,返回文件描述符f 46 mapped=(RECORD *)mmap(0,NRECORDS*sizeof(record),PROT_READ|PROT_WRITE,MAP_SHARED,f,0); 47 //创建指向0的内存,长度是100个指定长度,权限是读写内存段,flag取MAP_SHARED表示对内存段的修改保存到 48 //磁盘文件中,f就是和内存相关的描述符,偏移是0。总体来说是把文件转化为内存中的结构数组 49 mapped[43].integer=243;//设置结构数组中的第43个元素的第一个值 50 sprintf(mapped[43].string,"RECORD-%d",mapped[43].integer);//同样按照左对齐方式传递给结构数组第二个元素 51 52 msync((void *)mapped, NRECORDS*sizeof(record),MS_ASYNC);//把内存中的整段的修改按照异步写的方式写回到被映射的文件中 53 munmap((void *)mapped, NRECORDS*sizeof(record));//全部释放这个内存段 54 close(f);//关闭打开的文件描述符 55 56 exit(0); 57 }
如果是前面不使用mmap程序可以顺利执行,但是加上后面的代码时候,terminal会提示说:
段错误 (核心已转储)
参考文献:
Linux程序设计 Neil Matthew