日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學(xué)無先后,達(dá)者為師

網(wǎng)站首頁 編程語言 正文

C語言文件操作之fread函數(shù)詳解_C 語言

作者:韓曙亮 ? 更新時(shí)間: 2022-08-05 編程語言

前言

二進(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

欄目分類
最近更新