- 創建Makefile—
創建Makefile
testfile: main.c mystdio.c gcc -o $@ $^ .PHONY:clean clean: rm -f testfile
- mystdio.h ——接口的聲明,創建MY_FILE結構體,內部包含文件描述符fd,輸出緩沖區outputbuffer、flags刷新方法
—
通過C庫中fopen、fwrite、fclose接口的實現,設計自己的接口
—
—
—
3. mystdio.c —— 接口的實現1. MY_fopen的實現1.識別標志位
實現了讀、寫、追加方式
- 嘗試打開文件若想打開文件,需要調用open函數
—
—
若需要創建文件,則需調用第二個open函數。由于open中的mode參數受umask影響,所以設置一個默認的mode。若不需要創建文件,則調用第一個open函數。
- 給用戶返回MY_FILE對象,需要先創建對象
判斷對象是否創建成功,若失敗需要將文件關閉。
4.初始化MY_FILE對象 將結構體MY_FILE內部的fd賦值為open函數打開的返回值fd。刷新方法設置成行緩沖,outputbuffer緩沖區中全部初始化為0,current代表緩沖區中沒有數據。
5.返回打開的文件當關閉文件的時候,fclose(FILE*)將c語言當中的文件指針傳進來。當關閉文件的時候,C要自己幫助我們進行沖刷緩沖區。為了方便表述,在MY_FILE結構體添加current變量。
current代表下次寫入時應該寫入的位置,如outputbuffer中有5個字符,對應下標0 1 2 3 4,所以current代表下標5。
2.MY_close 的實現沖刷緩沖區自己實現一個fflush(刷新緩沖區),叫做MY_fflush
—
判斷緩沖區是否有數據,若有數據就刷新出去。
3. MY_fwrite的實現
緩沖區為ptr,單個單元的大小為size,nmemb代表想要寫入幾個單元,寫入對應的流中。實際上是往緩沖區里寫的。
1.緩沖區如果已經滿了,就直接寫入流中刷新流的緩沖區。
2.根據緩沖區剩余情況,進行拷貝 共分為兩種情況,若剩余空間足夠,則調用if語句,將用戶從ptr拷貝的數據全部拷貝給緩沖區,同時由于緩沖區加入user_size個字節,要更新current的位置。若剩余空間不足夠,則調用else語句,將從ptr拷貝的數據填滿剩余空間即可,同時由于緩沖區加入MY_size個字節,要更新current的位置。
通過調用writen代表實際寫了多少字節,為了充當最后的返回值。
- 開始計劃刷新
主要分為全刷新和行刷新兩種情況,其他不考慮。全刷新判斷緩沖區是否滿了,若滿了則直接刷新緩沖區。行刷新判斷是否遇見n,若遇見n則直接刷新緩沖區。
對之前內容清空為了防止出現每次打印都會有之前的內容情況,所以刷新之后要清空。
在這種情況下,之前的內容會被打印出來。
將current置為0后,下次寫入就可以覆蓋上次緩沖區內容。
- 整體代碼1. main.c
#include"mystdio.h" #include<string.h> #include<unistd.h> #define MYFILE "log.txt" int main() { MY_FILE*fp=MY_fopen(MYFILE,"w"); if(fp==NULL) return 1; const char*str="hello world"; int cnt=5; //操作文件 while(1) { char buffer[1024]; snprintf(buffer,sizeof(buffer),"%s:%dn",str,cnt--); size_t size=MY_fwrite(buffer,strlen(buffer),1,fp); sleep(1); printf("當前成功寫入:%lu個字節n",size); } MY_fclose(fp); return 0; }
- mystdio.h
#include<stdio.h> #define NUM 1024 #define BUFF_NONE 0x1 //表示無緩沖 #define BUFF_LINE 0x2 //行緩沖 #define BUFF_ALL 0x4 //全緩沖 typedef struct MY_FILE { int fd;//文件描述符 int flags;//刷新方法 char outputbuffer[1024];//輸出緩沖區 int current; }MY_FILE; MY_FILE *MY_fopen(const char *path, const char *mode);//自己寫fopen size_t MY_fwrite(const void *ptr, size_t size, size_t nmemb, MY_FILE *stream);//自己寫的fwrite int MY_fclose(MY_FILE *fp);//自己寫的fwrite int MY_fflush (MY_FILE*fp);//自己實現的緩沖區
- mystdio.c
#include"mystdio.h" #include<string.h> #include<sys> #include<sys> #include<fcntl.h> #include<stdlib.h> #include<unistd.h> #include<assert.h> MY_FILE*MY_fopen(const char *path, const char *mode)//自己寫fopen { int flag=0; if(strcmp(mode,"r")==0)//說明當前使用讀方式打開文件 flag |= O_RDONLY;//讀取 else if(strcmp(mode,"w")==0) flag |=(O_CREAT | O_WRONLY | O_TRUNC);//創建文件 以寫的方式打開文件 清空文件 else if(strcmp(mode,"a")==0) flag |=(O_CREAT | O_WRONLY | O_APPEND); //創建文件 以寫的方式打開文件 追加 else { //其他不考慮 } //2. 嘗試打開文件 mode_t m=0666; int fd=0; //flag代表模式 r w a if(flag & O_CREAT) fd=open(path,flag,m); else //說明不需要打開 fd=open(path,flag); if(fd<0) return NULL; //3. 給用戶返回MY_FILE對象,需要先創建對象 MY_FILE *mf = (MY_FILE*)malloc(sizeof(MY_FILE)); if(mf == NULL) { close(fd); return NULL; } //4. 初始化MY_FILE對象 mf->fd=fd;//將上述的fd傳入結構體的fd中 mf->flags=0; mf->flags=BUFF_LINE;//設置成行緩沖 memset(mf->outputbuffer,' 久久久精品波多野结衣| 亚洲一级Av无码毛片久久精品| 亚洲国产另类久久久精品黑人| 偷窥少妇久久久久久久久| 久久久久久久波多野结衣高潮 | 久久99精品久久久久久9蜜桃| 国产精品欧美久久久久天天影视| 国产精品VIDEOSSEX久久发布| 伊人久久五月天| 亚洲国产欧洲综合997久久| 99精品久久精品一区二区| 国产69精品久久久久99| 日本五月天婷久久网站| 免费精品99久久国产综合精品| 狠狠色伊人久久精品综合网| 久久天天躁狠狠躁夜夜不卡| 久久综合久久综合久久| 一本色道久久综合亚洲精品| 久久国产香蕉视频| 久久久久国产精品熟女影院| 久久乐国产精品亚洲综合| 久久亚洲私人国产精品vA| 欧美国产成人久久精品| 国产精品久久成人影院| 久久WWW免费人成一看片| 精品乱码久久久久久夜夜嗨| 99久久婷婷免费国产综合精品| 久久久久亚洲AV无码观看| 99久久久久| 久久天堂电影网| 99精品国产在热久久| 精品久久久久久亚洲精品| 狠狠色婷婷久久一区二区| 久久影视国产亚洲| 狠狠色伊人久久精品综合网| 久久99热精品| 国产精品久久免费| 99久久777色| 久久综合九色综合97_久久久| 久久电影网一区| 国产成人久久777777|