網(wǎng)站首頁 編程語言 正文
前言
二進(jìn)制文件讀寫兩個(gè)重要的函數(shù) , fread 和 fwrite , fread 用于讀取文件 , fwrite 用于寫出文件 ;
fread / fwrite 函數(shù) 既可以操作 二進(jìn)制文件 , 又可以操作 文本文件 ;
getc / putc 函數(shù) , fscanf / fprintf 函數(shù) , fgets / fgets 函數(shù) , 只能用于操作 文本文件 ;
一、fread 函數(shù)
fread 函數(shù)作用 : 從文件中讀取若干字節(jié)數(shù)據(jù)到內(nèi)存緩沖區(qū)中 ;
fread 函數(shù)原型 :
size_t fread( void *buffer, size_t size, size_t count, FILE *stream );
void *buffer 參數(shù) : 將文件中的二進(jìn)制數(shù)據(jù)讀取到該緩沖區(qū)中 ;
size_t size 參數(shù) : 讀取的 基本單元 字節(jié)大小 , 單位是字節(jié) , 一般是 buffer 緩沖的單位大小 ;
- 如果 buffer 緩沖區(qū)是 char 數(shù)組 , 則該參數(shù)的值是 sizeof(char) ;
- 如果 buffer 緩沖區(qū)是 int 數(shù)組 , 則該參數(shù)的值是 sizeof(int) ;
size_t count 參數(shù) : 讀取的 基本單元 個(gè)數(shù) ;
FILE *stream 參數(shù) : 文件指針 ;
size_t 返回值 : 實(shí)際從文件中讀取的 基本單元 個(gè)數(shù) ; 讀取的字節(jié)數(shù)是 基本單元數(shù) * 基本單元字節(jié)大小 ;
代碼示例 : 一次性讀滿整個(gè)緩沖區(qū) ;
#include <stdio.h> int main() { // 使用 "rb" 讀取二進(jìn)制方式打開文件 FILE *p = fopen("D:\\a.txt", "rb"); // 用于接收讀取數(shù)據(jù)的緩沖區(qū) char buffer[1024] = {0}; // buffer : 將文件讀取到內(nèi)存的位置 // sizeof(char) : 讀取的基本單元字節(jié)長(zhǎng)度 // sizeof(buffer) : 讀取的基本單元個(gè)數(shù), // 讀取字節(jié)個(gè)數(shù)是 sizeof(buffer) * sizeof(char) // p : 文件指針 fread(buffer, sizeof(char), sizeof(buffer), p); // 打印讀取的內(nèi)容 printf("buffer = %s\n", buffer); printf("Main End\n"); return 0; }
執(zhí)行結(jié)果 :
二、緩沖區(qū)受限的情況 ( 循環(huán)讀取文件 | feof 函數(shù)判定文件讀取完畢 )
假設(shè)緩沖區(qū)很小 , 文件很大 , 則需要循環(huán)讀取文件數(shù)據(jù) ;
使用 feof(p)
判定文件是否讀取完畢 , 如果返回 true 說明文件沒有讀取完畢 , 返回 false , 說明文件讀取完畢 ;
代碼示例 :
#include <stdio.h> int main() { // 使用 "rb" 讀取二進(jìn)制方式打開文件 FILE *p = fopen("D:\\a.txt", "rb"); // 用于接收讀取數(shù)據(jù)的緩沖區(qū) char buffer[4] = {0}; while(!feof(p)){ memset(buffer, 0, sizeof(buffer)); // buffer : 將文件讀取到內(nèi)存的位置 // sizeof(char) : 讀取的基本單元字節(jié)長(zhǎng)度 // sizeof(buffer) : 讀取的基本單元個(gè)數(shù), // 讀取字節(jié)個(gè)數(shù)是 sizeof(buffer) * sizeof(char) // p : 文件指針 fread(buffer, sizeof(char), sizeof(buffer), p); // 打印讀取的內(nèi)容 printf("buffer = %s\n", buffer); } printf("Main End\n"); return 0; }
執(zhí)行結(jié)果 : 讀取之后出現(xiàn)亂碼 , 這是由于每次讀取 10 10 10 字節(jié) , 但是字符串必須要以 ‘\0’ 進(jìn)行結(jié)尾 , 如果沒有 ‘\0’ 則會(huì)一直讀取直到出現(xiàn) ‘\0’ 字符串結(jié)尾位置 ;
三、處理亂碼問題
為了避免上述打印出現(xiàn)亂碼的情況 , char buffer[4] = {0};
準(zhǔn)備了 4 4 4 字節(jié)緩沖區(qū) , 每次只使用其中的 3 3 3 個(gè)字節(jié) , 這就能保證最后一個(gè)字節(jié)必定是 ‘\0’ , 打印時(shí)就不會(huì)出現(xiàn)亂碼 ;
代碼示例 :
#include <stdio.h> int main() { // 使用 "rb" 讀取二進(jìn)制方式打開文件 FILE *p = fopen("D:\\a.txt", "rb"); // 用于接收讀取數(shù)據(jù)的緩沖區(qū) char buffer[4] = {0}; while(!feof(p)){ memset(buffer, 0, sizeof(buffer)); // buffer : 將文件讀取到內(nèi)存的位置 // sizeof(char) : 讀取的基本單元字節(jié)長(zhǎng)度 // sizeof(buffer) : 讀取的基本單元個(gè)數(shù), // 讀取字節(jié)個(gè)數(shù)是 sizeof(buffer) * sizeof(char) // p : 文件指針 fread(buffer, sizeof(char), sizeof(buffer) - 1, p); // 打印讀取的內(nèi)容 printf("buffer = %s\n", buffer); } printf("Main End\n"); return 0; }
執(zhí)行結(jié)果 : 每次從文件中讀取 緩沖區(qū)字節(jié)數(shù) - 1 個(gè)字節(jié) , 則能完整的將文本打印出來 ;
四、記錄讀取的字節(jié)個(gè)數(shù)
fread 函數(shù)返回值表示讀取到的 基本單元 的個(gè)數(shù) , 如果設(shè)置了 1KB 的緩沖區(qū) , 但是文件中只有 5 字節(jié) , 則 fread 的返回值就是實(shí)際讀取到的數(shù)據(jù)個(gè)數(shù) ;
代碼示例 :
#include <stdio.h> int main() { // 使用 "rb" 讀取二進(jìn)制方式打開文件 FILE *p = fopen("D:\\a.txt", "rb"); // 用于接收讀取數(shù)據(jù)的緩沖區(qū) char buffer[1024] = {0}; // buffer : 將文件讀取到內(nèi)存的位置 // sizeof(char) : 讀取的基本單元字節(jié)長(zhǎng)度 // sizeof(buffer) : 讀取的基本單元個(gè)數(shù), // 讀取字節(jié)個(gè)數(shù)是 sizeof(buffer) * sizeof(char) // p : 文件指針 // 返回值 : fread 函數(shù)返回值表示讀取到的 基本單元 的個(gè)數(shù) size_t count = fread(buffer, sizeof(char), sizeof(buffer) - 1, p); // 打印讀取的內(nèi)容 printf("buffer = %s , read count = %u\n", buffer, count); printf("Main End\n"); return 0; }
執(zhí)行結(jié)果 :
五、讀取到 0 字節(jié)的情況
如果 基本單元 大小 4 4 4 字節(jié) , 文件中只有 3 3 3 字節(jié)數(shù)據(jù) , 則使用 fread 函數(shù)讀取文件 , 緩沖區(qū)設(shè)置 1KB , 則實(shí)際讀取到的基本單元個(gè)數(shù)是 0 0 0 ;
代碼示例 :
#include <stdio.h> int main() { // 使用 "rb" 讀取二進(jìn)制方式打開文件 FILE *p = fopen("D:\\a.txt", "rb"); // 用于接收讀取數(shù)據(jù)的緩沖區(qū) char buffer[1024] = {0}; // buffer : 將文件讀取到內(nèi)存的位置 // sizeof(char) : 讀取的基本單元字節(jié)長(zhǎng)度 // sizeof(buffer) : 讀取的基本單元個(gè)數(shù), // 讀取字節(jié)個(gè)數(shù)是 sizeof(buffer) * sizeof(char) // p : 文件指針 // 返回值 : fread 函數(shù)返回值表示讀取到的 基本單元 的個(gè)數(shù) size_t count = fread(buffer, sizeof(int), sizeof(buffer) - 1, p); // 打印讀取的內(nèi)容 printf("buffer = %s , read count = %u\n", buffer, count); printf("Main End\n"); return 0; }
執(zhí)行結(jié)果 :
六、讀取完畢的情況
如果文件已經(jīng)讀取完畢 , 不關(guān)閉文件 , 再次調(diào)用 fread 函數(shù)繼續(xù)讀取 , 則讀取到的 基本單元 個(gè)數(shù)是 0 0 0 ;
使用 feof(p)
判定文件是否讀取完畢 , 如果返回 true 說明文件沒有讀取完畢 , 返回 false , 說明文件讀取完畢 ;
代碼示例 :
#include <stdio.h> int main() { // 使用 "rb" 讀取二進(jìn)制方式打開文件 FILE *p = fopen("D:\\a.txt", "rb"); // 用于接收讀取數(shù)據(jù)的緩沖區(qū) char buffer[1024] = {0}; // buffer : 將文件讀取到內(nèi)存的位置 // sizeof(char) : 讀取的基本單元字節(jié)長(zhǎng)度 // sizeof(buffer) : 讀取的基本單元個(gè)數(shù), // 讀取字節(jié)個(gè)數(shù)是 sizeof(buffer) * sizeof(char) // p : 文件指針 // 返回值 : fread 函數(shù)返回值表示讀取到的 基本單元 的個(gè)數(shù) size_t count = fread(buffer, sizeof(char), sizeof(buffer) - 1, p); // 打印第一次讀取的內(nèi)容 printf("First fread : buffer = %s , read count = %u\n", buffer, count); count = fread(buffer, sizeof(int), sizeof(buffer) - 1, p); // 打印第二次讀取的內(nèi)容 printf("Second fread : buffer = %s , read count = %u\n", buffer, count); printf("Main End\n"); return 0; }
執(zhí)行結(jié)果 :
七、讀取文本文件 “\n” 與 讀取二進(jìn)制文件 “\r\n” 區(qū)別
以下區(qū)別只在 Windows 系統(tǒng)存在 , 在 Linux / Unix 中讀取文本數(shù)據(jù)與二進(jìn)制數(shù)據(jù)沒有區(qū)別 ;
使用 ‘rb’ 方式打開文件 , 讀取二進(jìn)制文件 , 然后調(diào)用 fread 函數(shù)讀取文件 ,
#include <stdio.h> int main() { // 使用 "rb" 讀取二進(jìn)制方式打開文件 FILE *p = fopen("D:\\a.txt", "rb"); // 用于接收讀取數(shù)據(jù)的緩沖區(qū) char buffer[1024] = {0}; // buffer : 將文件讀取到內(nèi)存的位置 // sizeof(char) : 讀取的基本單元字節(jié)長(zhǎng)度 // sizeof(buffer) : 讀取的基本單元個(gè)數(shù), // 讀取字節(jié)個(gè)數(shù)是 sizeof(buffer) * sizeof(char) // p : 文件指針 // 返回值 : fread 函數(shù)返回值表示讀取到的 基本單元 的個(gè)數(shù) size_t count = fread(buffer, sizeof(char), sizeof(buffer) - 1, p); // 打印第一次讀取的內(nèi)容 printf("fread : buffer = %s , read count = %u\n", buffer, count); // 逐個(gè)字節(jié)打印讀取出數(shù)據(jù)的 ASCII 碼 int i = 0; for(i = 0; i < count; i ++){ printf("buffer[%d] = %x\n", i, buffer[i]); } printf("Main End\n"); return 0; }
執(zhí)行結(jié)果 : 第 2 2 2 個(gè)索引讀取出來的值是 0xd 對(duì)應(yīng) ‘\r’ , 第 3 3 3 個(gè)值是 0xa 對(duì)應(yīng) ‘\n’ ;
注意 : 最后兩個(gè)字節(jié)是空行對(duì)應(yīng)的 “\r\n” ;
fread : buffer = ab cd , read count = 8 buffer[0] = 61 buffer[1] = 62 buffer[2] = d buffer[3] = a buffer[4] = 63 buffer[5] = 64 buffer[6] = d buffer[7] = a Main End
使用 ‘r’ 方式打開文件 , 讀取文本文件 , 然后調(diào)用 fread 函數(shù)讀取文件 ,
#include <stdio.h> int main() { // 使用 "rb" 讀取二進(jìn)制方式打開文件 FILE *p = fopen("D:\\a.txt", "r"); // 用于接收讀取數(shù)據(jù)的緩沖區(qū) char buffer[1024] = {0}; // buffer : 將文件讀取到內(nèi)存的位置 // sizeof(char) : 讀取的基本單元字節(jié)長(zhǎng)度 // sizeof(buffer) : 讀取的基本單元個(gè)數(shù), // 讀取字節(jié)個(gè)數(shù)是 sizeof(buffer) * sizeof(char) // p : 文件指針 // 返回值 : fread 函數(shù)返回值表示讀取到的 基本單元 的個(gè)數(shù) size_t count = fread(buffer, sizeof(char), sizeof(buffer) - 1, p); // 打印第一次讀取的內(nèi)容 printf("fread : buffer = %s , read count = %u\n", buffer, count); // 逐個(gè)字節(jié)打印讀取出數(shù)據(jù)的 ASCII 碼 int i = 0; for(i = 0; i < count; i ++){ printf("buffer[%d] = %x\n", i, buffer[i]); } printf("Main End\n"); return 0; }
執(zhí)行結(jié)果 : 第 2 2 2 個(gè)索引讀取出來的值是 0xa 對(duì)應(yīng) ‘\n’ ;
最后的空行只有一個(gè) ‘\n’ ;
fread : buffer = ab cd , read count = 6 buffer[0] = 61 buffer[1] = 62 buffer[2] = a buffer[3] = 63 buffer[4] = 64 buffer[5] = a Main End
總結(jié)?
原文鏈接:https://blog.csdn.net/shulianghan/article/details/117338844
相關(guān)推薦
- 2022-06-15 go語言數(shù)組及結(jié)構(gòu)體繼承和初始化示例解析_Golang
- 2022-08-02 使用shell讀取ini文件方法步驟_linux shell
- 2023-04-06 C語言中的多行輸入問題及說明_C 語言
- 2022-06-06 max-width: 100%和width:100%的區(qū)別
- 2022-04-19 運(yùn)行 npm run xxx 的時(shí)候都執(zhí)行了些什么
- 2022-07-04 PyCharm如何配置SSH和SFTP連接遠(yuǎn)程服務(wù)器_python
- 2022-12-21 Android?O對(duì)后臺(tái)Service限制詳解_Android
- 2023-01-26 Redis的數(shù)據(jù)復(fù)制過程詳解_Redis
- 最近更新
-
- window11 系統(tǒng)安裝 yarn
- 超詳細(xì)win安裝深度學(xué)習(xí)環(huán)境2025年最新版(
- Linux 中運(yùn)行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲(chǔ)小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎(chǔ)操作-- 運(yùn)算符,流程控制 Flo
- 1. Int 和Integer 的區(qū)別,Jav
- spring @retryable不生效的一種
- Spring Security之認(rèn)證信息的處理
- Spring Security之認(rèn)證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權(quán)
- redisson分布式鎖中waittime的設(shè)
- maven:解決release錯(cuò)誤:Artif
- restTemplate使用總結(jié)
- Spring Security之安全異常處理
- MybatisPlus優(yōu)雅實(shí)現(xiàn)加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務(wù)發(fā)現(xiàn)-Nac
- Spring Security之基于HttpR
- Redis 底層數(shù)據(jù)結(jié)構(gòu)-簡(jiǎn)單動(dòng)態(tài)字符串(SD
- arthas操作spring被代理目標(biāo)對(duì)象命令
- Spring中的單例模式應(yīng)用詳解
- 聊聊消息隊(duì)列,發(fā)送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠(yuǎn)程分支