C语言进阶⑱(文件上篇)(动态通讯录写入文件)(文件指针+IO流+八个输入输出函数)fopen+fclose(下)

简介: C语言进阶⑱(文件上篇)(动态通讯录写入文件)(文件指针+IO流+八个输入输出函数)fopen+fclose

4.3文本行输出函数 fputs

介绍:将字符串写入到指定的流 stream 中(不包括空字符)。适用于所有输出流。

代码演示:利用 fputstest2.txt 中随便写入几行数据:

 
#include <stdio.h>
int main() 
{
    FILE* pf = fopen("test2.txt", "w");
    if (pf == NULL) 
    {
        perror("fopen");
        return 1;
    }
    // 写文件 - 按照行来写
    fputs("abcdef\n", pf);//文件里也会显示换行效果
    fputs("123456", pf);
    // 关闭文件
    fclose(pf);
    pf = NULL;
    return 0;
}

4.4文本行输入函数 fgets

介绍:从指定的流 stream 读取一行,并把它存储在 string 所指向的字符串中,当读取(n-1)个字符时,或者读取到换行符、到达文件末尾时,它会停止,具体视情况而定。适用于所有输入流。


注意事项:假如 n 是100,读取到的就是99个字符(n-1),因为要留一个字符给\0。


代码演示:利用 fgets 读取 test2.txt 中的内容:

 
#include <stdio.h>
int main() 
{
    char arr[10] = "xxxxxx"; // 存放处
    FILE* pf = fopen("test2.txt", "r");
    if (pf == NULL) 
    {
        perror("fopen");
        return 1;
    }
    // 读文件 - 按照行来读
    fgets(arr, 4, pf);
    printf("%s\n", arr);
    fgets(arr, 4, pf);
    printf("%s\n", arr);
    // 关闭文件
    fclose(pf);
    pf = NULL;
    return 0;
}

4.5格式化输出函数 fprintf

介绍:fprintf 用于对格式化的数据进行写文件,发送格式化输出到流 stream 中。适用于所有输出流。

只需要在第一个函数参数放上一个文件指针,后面就和printf一样了。

代码演示:将结构体的三个数据利用 fprintf 写到 test3.txt 中:

 
#include <stdio.h>
struct student
{
    char name[10];
    int age;
    float x;
};
int main()
{
    struct student s = { "GR", 19, 3.14f };
    // 对格式化的数据进行写文件
    FILE* pf = fopen("test3.txt", "w");
    if (pf == NULL) 
    {
        perror("fopen");
        return 1;
    }
    // 写文件
    fprintf(pf, "%s %d %f", s.name, s.age, s.x);
 
    // 关闭文件
    fclose(pf);
    pf = NULL;
 
    return 0;
}

4.6格式化输入函数 fscanf

介绍:fscanf 用于对格式化的数据进行读取,从流 stream 读取格式化输入。适用于所有输入流。

只需要在第一个函数参数放上一个文件指针,后面就和scanf一样了。

代码演示:利用 fscanf 读取刚才 test3.txt 中的内容,并打印:

 
#include <stdio.h>
struct student 
{
    char name[10];
    int age;
    float x;
};
int main() 
{
    struct student s = { 0 }; // 存放处
 
    // 对格式化的数据进行写文件
    FILE* pf = fopen("test3.txt", "r");
    if (pf == NULL) 
    {
        perror("fopen");
        return 1;
    }
    // 读文件
    fscanf(pf, "%s %d %f",s.name, &(s.age), &(s.x)); //  s.name本身就是地址(不用&)
 
    // 将读到的数据打印
    printf("%s %d %f\n", s.name, s.age, s.x);
 
    // 关闭文件
    fclose(pf);
    return 0;
}

4.7二进制输出函数 fwrite

介绍:写一个数据到流中去,把 buffer 所指向的数组中的数据写入到给定流 stream 中。

const void *buffer : 指针指向要写出数据的内存首地址 ;

size_t size : 要写出数据的 基本单元 的字节大小 , 写出单位的大小 ;

size_t count : 要写出数据的 基本单元 的个数 ;

FILE *stream : 打开的文件指针 ;

返回值说明 : size_t 返回值返回的是实际写出到文件的 基本单元 个数 ;

创建一个 test4.txt,用 fwrite 写入一个数据到 text4.txt 中去:

 
#include <stdio.h>
struct student 
{
    char name[10];
    int age;
    float x;
};
int main() 
{
    struct student s = { "GR", 19, 3.14f }; // 存放处
 
    FILE* pf = fopen("test4.txt", "w");
    if (pf == NULL) 
    {
        perror("fopen");
        return 1;
    }
    //写文件
    fwrite(&s, sizeof(struct student), 1, pf);
 
    // 关闭文件
    fclose(pf);
    return 0;
}

运行后打开文件后我们发现只有字母看得懂,后面是什么我们看不懂。

① 我们刚才用的都是文本编译器,文本编译器打开二进制形式的文件完全是两种状态。

② 因为字符串以文本形式写进去和以二进制形式写进去是一样的,但是对于整数、浮点数等

来说就不一样了,文本形式写入和二进制形式写入完全是两个概念。

(那么该怎么读呢,我们来看下面的 fread 函数)

4.8二进制输入函数 fread

介绍:从流中读取,从给定流 stream 读取数据到 buffer 所指向的数组中。参数和fwrite一样

fread 读取 text4.txt 中的二进制数据:

 
#include <stdio.h>
struct student 
{
    char name[10];
    int age;
    float x;
};
int main() 
{
    struct student s = { 0 }; // 存放处
 
    FILE* pf = fopen("test4.txt", "r");
    if (pf == NULL) 
    {
        perror("fopen");
        return 1;
    }
    // 读文件
    fread(&s, sizeof(struct student), 1, pf);
 
    // 将读到的数据打印
    printf("%s %d %f", s.name, s.age, s.x);
 
    // 关闭文件
    fclose(pf);
    return 0;
}

总结: fwritefread 是一对,fwrire 写进去用 fread 读。


把动态通讯录写入文件

首先想到,在上一篇动态通讯录基础性应在退出销毁通讯录函数前设置一个保存通讯录到文件的函数。


那么自然,我们也需要打开通讯录的时候自动读入那个文件,每次打开通讯录我们都会运行init_contact函数,因此我们可以在init_contact函数中利用fread函数读入二进制文件。

写一个加载联系人函数:

//加载联系人函数
void load_contact(contact* p)
{
    FILE* pf = fopen("contact.dat", "r");
    if (pf == NULL)
    {
        perror("load_contact");
        return;
    }
    //读文件
    peoinfo tmp = { 0 };
    while (fread(&tmp, sizeof(peoinfo), 1, pf))
    {
        //是否需要增容
        check_capacity(p);
        p->data[p->sz] = tmp;
        p->sz++;
    }
    
    //关闭文件
    fclose(pf);
    pf = NULL;
}

把之前add函数中负责扩容的部分单独提取出来封装为check_capacity函数,

因为负责扩容的部分利用if语句判断了当前是否需要扩容,这是为了读入文件做的准备。


读文件这里有一个难点,我们并不知道这个二进制文件里头有多少个通讯录成员,

可我们是动态扩容的,无法确定当前容量能不能撑得下这些通讯录成员。


可以利用fread函数的返回值,其返回值表示从二进制文件中读入了多少个元素,


利用一个while循环,每次只读一个通讯录成员信息进来,


循环条件为fread函数的返回值不为0(也就是还没读完),那如果容量不够怎么办?


利用check_capacity函数,每次执行时,先执行check_capacity函数,


如果当前容量不足则扩容,然后再利用fread函数从二进制文件中读入通讯录成员的信息。


代码:

test.c

 
//通讯录-静态版本
//1.通讯录中能够存放1000个人的信息
//每个人的信息:
//名字+年龄+性别+电话+地址
//2. 增加人的信息
//3. 删除指定人的信息
//4. 修改指定人的信息
//5. 查找指定人的信息
//6. 排序通讯录的信息
//
//版本2:
//动态增长的版本
//1.通讯录初始化后,能存放3个人的信息
//2.当我们空间的存放满的时候,我们增加2个信息
//3+2+2+2+...
 
//
//版本3
//当通讯录退出的时候,把信息写到文件
//当通讯录初始化的时候,加载文件的信息到通讯录中
//
 
#include"contact.h"
 
void menu()
{
    printf("****************************************\n");
    printf("******     1.add      2.del     ********\n");
    printf("******     3.search   4.modify  ********\n");
    printf("******     5.sort     6.print   ********\n");
    printf("******     0.exit               ********\n");
    printf("****************************************\n");
}
enum option
{
    EXIT,
    ADD,
    DEL,
    SEARCH,
    MODIFY,
    SORT,
    PRINTF
};
int main()
{
    int input = 0;
    //创建通讯录  info(信息)
    contact con;
    //初始化通讯录函数
    init_contact(&con);
    do
    {
        menu();
        printf("请选择:");
        scanf("%d", &input);
        switch (input)
        {
        case ADD:
            add_contact(&con);
            break;
        case DEL:
            del_contact(&con);
            break;
        case SEARCH:
            find_contact(&con);
            break;
        case MODIFY:
            modify_contact(&con);
            break;
        case SORT:
            sort_contact(&con);
            break;
        case PRINTF:
            print_contact(&con);
            break;
        case EXIT:
            save_contact(&con);
            destroy_contact(&con);
            printf("退出程序\n");
            break;
        default:
            printf("选择错误,重新选择\n");
            break;
        }
    } while (input);
    return 0;
}

contact.h

 
#include<stdio.h>
#include<string.h>
#include<stdlib.h>//qsort,perror,动态内存开辟函数
 
#define MAX_NAME 20
#define MAX_SEX 10
#define MAX_TELE 20
#define MAX_ADDR 30
#define DEFAULT_SZ 3 //default 默认
#define INC_SZ 2 //sz增量
//类型的定义
typedef struct peoinfo
{
    char name[MAX_NAME];
    char sex[MAX_SEX];
    int age;
    char tele[MAX_TELE];
    char addr[MAX_ADDR];
}peoinfo;
 
静态版本通讯录
//typedef struct contact
//{
//    peoinfo data[MAX];//存放添加进来的人的信息     
//    int sz;//记录的是当前通讯录中有效信息的个数
//}contact;
 
//动态版本通讯录
typedef struct contact
{
    peoinfo* data;//指向动态开辟的空间,存放添加进来的人的信息
    int sz;//记录的是当前通讯录中有效信息的个数
    int capacity;//记录当前通讯录的最大容量,方便增容
}contact;
 
//初始化通讯录函数
void init_contact(contact* p);
 
//是否需要增容函数
void check_capacity(contact* p);
 
//加载联系人函数
void load_contact(contact* p);
 
//增加联系人函数
void add_contact(contact* p);
 
//保存联系人函数
void save_contact(contact* p);
 
//销毁联系人函数
void destroy_contact(contact* p);
 
//打印联系人函数
void print_contact(const contact* p);
 
//删除联系人函数
void del_contact(contact* p);
 
//查找联系人函数
void find_contact(const contact* p);
 
//修改联系人函数
void modify_contact(contact* p);
 
//排序联系人函数
void sort_contact(contact* p);

contact.c

 
#include"contact.h"
 
初始化通讯录函数   静态版本
//void init_contact(contact* p)
//{
//    p->sz = 0;
//    memset(p->data, 0, sizeof(p->data));
//}
 
//初始化通讯录函数    动态版本
void init_contact(contact* p)
{
    p->data = (peoinfo*)calloc(DEFAULT_SZ, sizeof(peoinfo));
    if (p->data == NULL)
    {
        perror("init_contact");
        return;
    }
    p->sz = 0;//初始化后默认是0
    p->capacity = DEFAULT_SZ;
 
    //加载文件(读文件)
    load_contact(p);
}
//是否需要增容函数
void check_capacity(contact* p)
{
    if (p->sz == p->capacity)
    {
        peoinfo* tmp = (peoinfo*)realloc(p->data, (p->capacity + INC_SZ) * sizeof(peoinfo));
        if (tmp == NULL)
        {
            perror("add_contact");
            printf("增容失败\n");
            return;
        }
        p->data = tmp;
        p->capacity += INC_SZ;
        printf("增容成功\n");
    }
}
//加载联系人函数
void load_contact(contact* p)
{
    FILE* pf = fopen("contact.dat", "r");
    if (pf == NULL)
    {
        perror("load_contact");
        return;
    }
    //读文件
    peoinfo tmp = { 0 };
    while (fread(&tmp, sizeof(peoinfo), 1, pf))
    {
        //是否需要增容
        check_capacity(p);
        p->data[p->sz] = tmp;
        p->sz++;
    }
    
    //关闭文件
    fclose(pf);
    pf = NULL;
}
 
增加联系人函数      静态版本
//void add_contact(contact* p)
//{
//    if (p->sz == MAX)
//    {
//        printf("通讯录已满,无法添加");
//        return;
//    }
//    printf("请输入要添加人的姓名:");
//    scanf("%s", p->data[p->sz].name);
//    printf("请输入要添加人的性别:");
//    scanf("%s", p->data[p->sz].sex);
//    printf("请输入要添加人的年龄:");
//    scanf("%d", &p->data[p->sz].age);//只用年龄不是数组,要取地址
//    printf("请输入要添加人的电话:");
//    scanf("%s", p->data[p->sz].tele);
//    printf("请输入要添加人的住址:");
//    scanf("%s", p->data[p->sz].addr);
//
//    p->sz++;
//    printf("添加成功\n");
//}
 
//增加联系人函数      动态版本
void add_contact(contact* p)
{
    check_capacity(p);
 
    printf("请输入要添加人的姓名:");
    scanf("%s", p->data[p->sz].name);
    printf("请输入要添加人的性别:");
    scanf("%s", p->data[p->sz].sex);
    printf("请输入要添加人的年龄:");
    scanf("%d", &p->data[p->sz].age);//只用年龄不是数组,要取地址
    printf("请输入要添加人的电话:");
    scanf("%s", p->data[p->sz].tele);
    printf("请输入要添加人的住址:");
    scanf("%s", p->data[p->sz].addr);
 
    p->sz++;
    printf("添加成功\n");
}
 
//保存联系人函数
void save_contact(contact* p)
{
    FILE* pf = fopen("contact.dat", "w");
    if (pf == NULL)
    {
        perror("save_contact");
        return;
    }
    //写文件
    for (int i = 0;i < p->sz;i++)
    {
        fwrite(p->data + i, sizeof(peoinfo), 1, pf);
    }
    //关闭文件
    fclose(pf);
    pf = NULL;
}
 
//销毁联系人函数
void destroy_contact(contact* p)
{
    free(p->data);
    p->data = NULL;
    p->sz = 0;
    p->capacity = 0;
}
 
//打印联系人函数
void print_contact(const contact* p)
{
    if (p->sz == 0)
    {
        printf("通讯录为空,无法打印\n");
    }
    else
    {
        printf("%-10s  %-10s  %-10s  %-15s  %-10s\n", "姓名", "性别", "年龄", "电话", "住址");
        for (int i = 0;i < p->sz;i++)
        {
            printf("%-10s  %-10s  %-10d  %-15s  %-10s\n",
                p->data[i].name,
                p->data[i].sex,
                p->data[i].age,
                p->data[i].tele,
                p->data[i].addr);
        }
    }
}
 
//查找函数的一部分,删除和修改也要用,只放在这就行
static int find_by_name(contact* p, char name[])
{
    for (int i = 0;i < p->sz;i++)
    {
        if (strcmp(p->data[i].name,name)==0)
        {
            return i;
        }
    }
    return -1;
}
 
//删除联系人函数
void del_contact(contact* p)
{
    if (p->sz == 0)
    {
        printf("通讯录为空,无法删除\n");
    }
    else
    {
        char name[MAX_NAME] = { 0 };
        printf("请输入要删除人的名字:");
        scanf("%s", name);
        int pos = find_by_name(p, name);
        if (pos == -1)
        {
            printf("要删除的人不存在\n");
        }
        else
        {
            for (int i = pos;i < p->sz - 1;i++)
            {
                p->data[i] = p->data[i + 1];
            }
            p->sz--;
            printf("删除成功\n");
        }
    }
}
 
//查找联系人函数
void find_contact(const contact* p)
{
    if (p->sz == 0)
    {
        printf("通讯录为空,无法查找\n");
    }
    else
    {
        char name[MAX_NAME] = { 0 };
        printf("请输入要查找人的名字:");
        scanf("%s", name);
        int pos = find_by_name(p, name);
        if (pos == -1)
        {
            printf("要查找的人不存在\n");
        }
        else
        {
            printf("%-10s  %-10s  %-10s  %-15s  %-10s\n", "姓名", "性别", "年龄", "电话", "住址");
            printf("%-10s  %-10s  %-10d  %-15s  %-10s\n",
                p->data[pos].name,
                p->data[pos].sex,
                p->data[pos].age,
                p->data[pos].tele,
                p->data[pos].addr);
        }
    }
}
 
//修改联系人函数
void modify_contact(contact* p)
{
    if (p->sz == 0)
    {
        printf("通讯录为空,无法查找\n");
    }
    else
    {
        char name[MAX_NAME] = { 0 };
        printf("请输入要修改人的姓名:");
        scanf("%s", name);
        int pos = find_by_name(p, name);
        if (pos == -1)
        {
            printf("要修改的人不存在\n");
        }
        else
        {
            printf("以下是输入要修改人的新信息:\n");
            printf("请输入姓名:");
            scanf("%s", p->data[pos].name);
            printf("请输入性别:");
            scanf("%s", p->data[pos].sex);
            printf("请输入年龄:");
            scanf("%d", &p->data[pos].age);
            printf("请输入电话:");
            scanf("%s", p->data[pos].tele);
            printf("请输入住址:");
            scanf("%s", p->data[pos].addr);
 
            printf("修改成功\n");
        }
    }
}
 
//排序联系人函数
int cmp(void* e1, void* e2)
{
    return strcmp(((peoinfo*)e1)->name, ((peoinfo*)e2)->name);
}
void sort_contact(contact* p)
{
    if (p->sz == 0)
    {
        printf("通讯录为空,无法排序\n");
    }
    else
    {
        qsort(p->data, p->sz, sizeof(peoinfo), cmp);
        printf("按照姓名排序成功\n");
    }
}
 
void sort_contact2(contact* p)//冒泡,比较麻烦,就用qsort了
{
    if (p->sz == 0)
    {
        printf("通讯录为空,无法排序\n");
    }
    else
    {
        for (int i = 0; i < p->sz; i++)
        {
            for (int j = 0; j < p->sz - i - 1; j++)
            {
                if (strcmp((p->data[j].name), (p->data[j + 1].name)) > 0)
                {
                    peoinfo data = p->data[j];
                    p->data[j] = p->data[j + 1];
                    p->data[j + 1] = data;
                }
            }
        }
        printf("按照姓名排序成功\n");
    }
}

本篇完。


328f9598ffb27e2d11bb887142d70c71.gif

目录
相关文章
|
2天前
|
算法 Java 程序员
面向对象编程(OOP)通过对象组合构建软件,C语言虽是过程式语言,但可通过结构体、函数指针模拟OOP特性
【6月更文挑战第15天】面向对象编程(OOP)通过对象组合构建软件,C语言虽是过程式语言,但可通过结构体、函数指针模拟OOP特性。封装可使用结构体封装数据和方法,如模拟矩形对象。继承则通过结构体嵌套实现静态继承。多态可通过函数指针模拟,但C不支持虚函数表,实现复杂。C语言能体现OOP思想,但不如C++、Java等语言原生支持。
16 7
|
3天前
|
安全 C语言
【C语言基础】:内存操作函数
【C语言基础】:内存操作函数
|
3天前
|
编译器 C语言 C++
【C语言基础】:字符函数和字符串函数-2
【C语言基础】:字符函数和字符串函数
|
3天前
|
C语言
【C语言基础】:字符函数和字符串函数-1
【C语言基础】:字符函数和字符串函数
|
3天前
|
C语言
【C语言基础】:深入理解指针(终篇)
【C语言基础】:深入理解指针(终篇)
|
3天前
|
存储 C语言 C++
【C语言基础】:深入理解指针(三)
【C语言基础】:深入理解指针(三)
|
3天前
|
安全 程序员 编译器
【C语言基础】:深入理解指针(二)
【C语言基础】:深入理解指针(二)
【C语言基础】:深入理解指针(二)
|
3天前
|
存储 编译器 C语言
【C语言基础】:深入理解指针(一)
【C语言基础】:深入理解指针(一)
TU^
|
3天前
|
编译器 程序员 Serverless
C语言之函数
C语言之函数
TU^
4 0
TU^
|
3天前
|
机器学习/深度学习 C语言
C语言之函数递归
C语言之函数递归
TU^
7 1