利用文件流实现通讯录

之前呢,我已经写过两种形式的通讯录:
静态通讯录:http://blog.csdn.net/quinn0918/article/details/71937767
动态通讯录:http://blog.csdn.net/quinn0918/article/details/72182598

fwrite函数原型:size_ t fwrite(const void* buffer, size_ t size, size_t count, FILE* stream)
函数功能:fwrite是C语言函数,指向文件写入一个数据块。
注意:这个函数以二进制形式对文件进行操作,不局限于文本文件
返回值:返回实际写入的数据块数目
(1)buffer:是一个指针,对fwrite来说,是要获取数据的地址;
(2)size:要写入内容的单字节数;
(3)count:要进行写入size字节的数据项的个数;
(4)stream:目标文件指针;
(5)返回实际写入的数据项个数count。
说明:写入到文件的哪里? 这个与文件的打开模式有关,如果是w+,则是从file pointer指向的地址开始写,替换掉之后的内容,文件的长度可以不变,stream的位置移动count个数;如果是a+,则从文件的末尾开始添加,文件长度加大。

fread 函数原型:size_ t fread ( void *buffer, size_ t size, size_t count, FILE *stream)
函数功能:fread是一个函数。从一个文件流中读数据,最多读取count个元素,每个元素size字节,如果调用成功返回实际读取到的元素个数,如果不成功或读到文件末尾返回 0。

代码如下:
linkman.h

#ifndef __LINKMAN_H__
#define __LINKMAN_H__

#include<stdio.h>
#include<windows.h>
#include<string.h>
#pragma warning (disable:4996)

#define MAX_NAME 20
#define MAX_SEX 10
#define MAX_TEL 12
#define MAX_ADDR 50
#define INIT 2
#define INC 3

typedef struct LINKMAN//建立结构体,存放联系人信息
{
    char name[20];
    char sex[10];
    int age;
    int tel[12];
    char addr[50];
}LINKMAN;

typedef struct Statis //把通讯录和人员统计放在结构体内
{
    LINKMAN *num;
    int info;//容量
    int count;//个数
}Statis;

void init_linkman(Statis *p);//初始化
void Add_linkman(Statis *p);// 添加联系人信息
void Dele_linkman(Statis *p);//删除指定联系人信息
void Find_linkman(Statis *p);//查找指定联系人信息
void Revise_linkman(Statis *p);//修改指定联系人信息
void Display_linkman(Statis *p);//显示所有联系人信息
void Empty_linkman(Statis *p);//清空所有联系人 
void sort_linkman(Statis *p);//以名字排序所有联系人
void Free_data(Statis *p);//释放内存
void SaveData(Statis *p);//把数据保存到文件内
void DownLoad(Statis *p);//从文件中读取数据

#endif

test.c

#include "linkman.h"
Statis sta;
void menu()//菜单
{
    printf("***********************************\n");
    printf("********1.添加联系人信息***********\n");
    printf("********2.删除指定联系人信息*******\n");
    printf("********3.查找指定联系人信息*******\n");
    printf("********4.修改指定联系人信息*******\n");
    printf("********5.显示所有联系人信息*******\n");
    printf("********6.清空所有联系人***********\n");
    printf("********7.以名字排序所有联系人*****\n");
    printf("**************按0退出程序**********\n");
    printf("***********************************\n");

}

void test()
{
    int i = 0;
    do
    {
        menu();
        printf("请输入你要进行的选项:");
        scanf("%d", &i);
        switch (i)
        {
        case 1:
            Add_linkman(&sta);
            break;
        case 2:
            Dele_linkman(&sta);
            break;
        case 3:
            Find_linkman(&sta);
            break;
        case 4:
            Revise_linkman(&sta);
            break;
        case 5:
            Display_linkman(&sta);
            break;
        case 6:
            Empty_linkman(&sta);
            break;
        case 7:
            sort_linkman(&sta);
            break;
        case 0:
            SaveData(&sta);
            Free_data(&sta);
            exit(0);
            break;
        default:
            printf("输入错误\n");
            break;
        }
    } while (i);
}


int main()
{
    init_linkman(&sta);
    test();

    system("pause");
    return 0;
}

linkman.c

#include "linkman.h"


void init_linkman(Statis *p)//初始化数组
{
    p->count = 0;
    p->info = INIT;
    p->num = (LINKMAN *)malloc(INIT*sizeof(LINKMAN));
    if (p->num == NULL)
    {
        printf("out of menory");
        exit(1);
    }
    else
    {
        DownLoad(p);
    }
}
void check(Statis *p)//检测动态内存分配
{
    if (p->info == p->count)
    {
        LINKMAN *tmp = (LINKMAN *)realloc(p->num, (p->info + INC)*sizeof(LINKMAN));
        //当实际联系人个数和初始化的容量相等时用realloc增容  
        if (tmp == NULL)
        {
            printf("out of menory\n");
            exit(1);
        }
        else
        {
            p->num = tmp;
            p->info += INC;
        }
    }
}
int Find(Statis *p, char *pname)//对于一个联系人是否存在封装一个函数,在后面的修改、删除、查找可以用到
{
    int i = 0;
    for (i = 0; i < (p->count); i++)
    {
        if (strcmp(p->num[i].name, pname) == 0)
            return i;

    }
    return -1;
}

void menu1()//修改联系人时所用到的菜单
{
    printf("*********************\n");
    printf("****1.姓名*2.性别****\n");
    printf("****3.年龄*4.电话****\n");
    printf("****5.地址*6.返回****\n");
    printf("*********************\n");
}

void Add_linkman(Statis *p)//添加联系人
{
    check(p);
    printf("请输入你要添加的联系人姓名:\n");
    scanf("%s", p->num[p->count].name);
    printf("请输入你要添加的联系人性别:\n");
    scanf("%s", p->num[p->count].sex);
    printf("请输入你要添加的联系人年龄:\n");
    scanf("%d", &(p->num[p->count].age));
    printf("请输入你要添加的联系人电话:\n");
    scanf("%s", p->num[p->count].tel);
    printf("请输入你要添加的联系人地址:\n");
    scanf("%s", p->num[p->count].addr);
    if ((p->count) > 1000)
    {
        printf("联系人上限\n");
    }
    else
    {
        printf("添加成功\n");
        p->count++;
    }
}

void Dele_linkman(Statis *p)//删除联系人
{
    char name[20] = { 0 };
    int result = 0;
    int n = 0;
    int j = 0;
    printf("请输入要删除人的姓名:\n");
    scanf("%s", name);
    result = Find(p, name);
    if (result != -1)
    {
        printf("你是否要删除该联系人?\n");
        printf("删除请按1,不删除请按0\n");
        scanf("%d", &n);
        if (n == 1)
        {
            for (j = result; j < (p->count)-1; j++)
            {
                p->num[j] = p->num[j + 1];
            }
            p->count --;
            printf("删除成功\n");
        }
        else
        {
            printf("删除失败\n");
        }
    }
    else
    {
        printf("你要删除的联系人不存在\n");
    }
}

void Find_linkman(Statis *p)//查找联系人
{
    char name[20] = { 0 };
    int result = 0;
    printf("请输入要查找联系人的姓名:\n");
    scanf("%s", name);
    result = Find(p, name);
    if (result != -1)
    {
        printf("姓名:%s\n", p->num[result].name);
        printf("性别:%s\n", p->num[result].sex);
        printf("年龄:%d\n", p->num[result].age);
        printf("电话:%s\n", p->num[result].tel);
        printf("地址:%s\n", p->num[result].addr);
    }
    else
    {
        printf("你要查找的联系人不存在\n");
    }
}

void Revise_linkman(Statis *p)//修改联系人
{
    char name[20] = { 0 };
    int result = 0;
    printf("请输入你要修改联系人的姓名:\n");
    scanf("%s", name);
    result = Find(p, name);
    if (result != -1)
    {
        printf("姓名:%s\n", p->num[result].name);
        printf("性别:%s\n", p->num[result].sex);
        printf("年龄:%d\n", p->num[result].age);
        printf("电话:%s\n", p->num[result].tel);
        printf("地址:%s\n", p->num[result].addr);
        int i = 0;
        do
        {
            menu1();
            printf("输入你要修改的选项:\n");
            scanf("%d", &i);
            switch (i)
            {
            case 1:
                printf("请把姓名修改成:");
                scanf("%s", p->num[result].name);
                break;
            case 2:
                printf("请把性别修改成:");
                scanf("%s", p->num[result].sex);
                break;
            case 3:
                printf("请把年龄修改成:");
                scanf("%d", &(p->num[result].age));
                break;
            case 4:
                printf("请把电话修改成:");
                scanf("%s", p->num[result].tel);
                break;
            case 5:
                printf("请把地址修改成:");
                scanf("%s", p->num[result].addr);
                break;
            case 0:
                break;
            default:
                printf("输入错误");
                break;
            }
        } while (i);
    }
    else
    {
        printf("你要修改的联系人不存在\n");
    }
}

void Display_linkman(Statis *p)//打印所有联系人信息
{
    int i = 0;
    printf("输出所有人的信息:\n");
    printf("%10s%7s%6s%8s%10s\n", "名字", "性别", "年龄", "电话", "住址");
    for (i = 0; i <(p->count); i++)
    {
        printf("%11s", p->num[i].name);
        printf("%5s", p->num[i].sex);
        printf("%5d", p->num[i].age);
        printf("%10s", p->num[i].tel);
        printf("%12s", p->num[i].addr);
        printf("\n");
    }
}

void Empty_linkman(Statis *p)//清空联系人
{
    p->count = 0;
}

void sort_linkman(Statis *p)//以名字排序所有联系人(冒泡)
{
    int i = 0;
    int j = 0;
    for (i = 0; i < p->count - 1; i++)
    for (j = 0; j < p->count - 1 - i; j++)
    {
        if (strcmp(p->num[j].name, p->num[j + 1].name)>0)
        {
            LINKMAN tmp;
            tmp = p->num[j];
            p->num[j] = p->num[j + 1];
            p->num[j + 1] = tmp;
        }
    }
}

void Free_data(Statis *p)
{
    free(p->num);
    p->info = 0;
    p->count = 0;
}
void SaveData(Statis *p)//保存数据到文件流 ,SaveData 的调用放在退出程序之前
{
    int i = 0;
    FILE *pf = fopen("tongxunlu.txt", "w");
    if (pf == NULL)
    {
        perror("open file for write");
        exit(1);
    }
    for (i = 0; i < p->count; i++)
    {
        fwrite(&(p->num[i]), sizeof(LINKMAN), 1, pf);
    }
    fclose(pf);
}


void DownLoad(Statis *p)//从文件里读取数据  
{  
    int i = 0;
    LINKMAN tmp = { 0 };
    FILE *pf = fopen("tongxunlu.txt", "r");
    if (pf == NULL)
    {
        perror("open file for read");
        exit(1);
    }

    while (fread(&tmp, sizeof (LINKMAN), 1, pf))
    {
        check(p);
        p->num[i] = tmp;
        i++;
        p->count++;//每读入一个数据,当前数据数目count就增加一个;  
    }
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值