通讯录管理系统(C语言版:源码+详解)

在学习C语言的过程中,为了检验C语言结构体学的如何,我们利用结构体的知识来做一个通讯录管理系统。


一.通讯录框架信息

构建通讯录管理系统我们所需要知道的:

1.人的信息:姓名,年龄,性别,地址,电话

2.通讯录中可以存放多少人的信息(例:100人)

3.通讯录可以使用的功能(这里列举几项):

->1.增加联系人

->2.删除联系人

->3.查找联系人

->4.显示通讯录

->5.修改指定联系人

->6.清空通讯录

二.框架的构建

1.创建菜单

为了将通讯录的功能全部的展现出来以及更好的管理通讯录系统,这里我们创建了菜单,使用者可以根据需求来使用不同的功能

void menu()
{
    printf("*************************\n");
    printf("****  通讯录管理系统   ****\n");
    printf("*** 1.Add     2.Del   ***\n");
    printf("*** 3.Search  4.Modify **\n");
    printf("*** 5.Show    6.Sort  ***\n");
    printf("*** 7.empty   8.exit  ***\n");
    printf("*************************\n");
}

2.成员信息的创建

 因为我们需要创建成员的信息和通讯录中成员的数量,所以我们需要两个结构体来创建:一个是存放人的信息,另一个是当前已放信息的个数

typedef struct PoeInfo  //通讯录成员的信息
{
    char name[NAME_MAX];
    int age;
    char sex[SEX_MAX];
    char addr[ADDR_MAX];
    char tele[TELE_MAX];
}PeoInfo;

静态版本:

typedef struct contast  
{
    PeoInfo data[MAX];  //存放人的信息
    int sz;             //当前已放信息的个数
}contast;

动态版本:

typedef struct contast
{
    PeoInfo* data;   //存放人的信息
    int sz;          //存放信息的个数
    int capacity;    //容量
}contast;

3.菜单功能声明

//增加联系人
void AddContast(contast* pc);

//删除联系人
void DelContast(contast* pc);

//查找指定联系人的信息
void SearchContast(contast* pc);

//修改指定联系人的信息
void ModifyContast(contast* pc);

//显示所有联系人的信息
void ShowContast(contast* pc);

//给通讯录排序
void SortContast(contast* pc);

//清空通讯录
void EmptyContast(contast* pc);

动态版本:新加了通讯录销毁功能(释放向系统申请的空间)

//销毁通讯录
void DestoryContast(contast* pc);  

三.通讯录功能的实现

1.初始化通讯录

void InitContast(contast* pc)
{
    assert(pc);
    pc->sz = 0;
    memset(pc->data, 0, sizeof(pc->data));
}

动态版本:

void InitContast(contast* pc)
{
    assert(pc);
    pc->sz = 0;
    PeoInfo* ptr = (PeoInfo*)calloc(DEFAULT_SZ, sizeof(PeoInfo));
    if(ptr == NULL)
    {
       printf("%s", strerror(errno));
       return;    
    }
    pc->data = ptr;
    pc->capacity = DEFAULT_SZ;
}

2.增加联系人

void AddContast(contast* pc)
{
    assert(pc);
    if(pc->sz == MAX)
    { 
       printf("通讯录已满,无法添加\n");
       return;
    }
      printf("请输入姓名:>");
      scanf("%s", pc->data[pc->sz].name );
      printf("请输入年纪:>");
      scanf("%d", &(pc->data[pc->sz].age) );
      printf("请输入性别:>");
      scanf("%s", pc->data[pc->sz].sex );
      printf("请输入地址:>");
      scanf("%s", pc->data[pc->sz].addr );
      printf("请输入电话号:>");
      scanf("%s", pc->data[pc->sz].tele );
      pz->sz++;
}

动态版本:

//增加容量
void check_capacity(contast* pc)
{
    assert(pc);
    if(pc->sz == pc->capacity)
    {
       PeoInfo* ptr = (*PeoInfo)realloc(pc->data, (capacity + DEFAULT_SZ) + sizeof(PeoInfo));
       if(ptr == NULL)
       {
          perror("cheak_capacity::realloc");
          return;
       }
       pc->data = ptr;
       pc->capacity += 2;
       printf("增容成功\n");
    }
}

void AddContast(contast* pc)
{
    assert(pc);
    check_capacity(pc);
    printf("请输入姓名:>");
    scanf("%s", pc->data[pc->sz].name );
    printf("请输入年纪:>");
    scanf("%d", &(pc->data[pc->sz].age) );
    printf("请输入性别:>");
    scanf("%s", pc->data[pc->sz].sex );
    printf("请输入地址:>");
    scanf("%s", pc->data[pc->sz].addr );
    printf("请输入电话号:>");
    scanf("%s", pc->data[pc->sz].tele );
    pz->sz++;
}

3.删除指定联系人

(1).寻找该成员

int FindByName(contast* ps, char* p)
{
    assert(ps && p);
    int i = 0;
    for(; i < ps->sz; i++)
    {
        if(strcmp(ps->data[i].name, p) == 0)
           return i;
    }
    return -1;
}

(2).删除该成员

void DelContast(contast* pc)
{
    assert(pc);
    if(pc->sz == 0)
    {
       printf("没有该成员\n");
       return;
    }
    char name[NAME_MAX] = {0};
    int del = FindByName(pc->data, name);
    if(del == -1)
    {
       printf("没有该成员\n");
       return;
    }
    int i = 0;
    for(i = del; i < pc->sz - 1; i++)
    {
        pc->data[i] = pc->data[i + 1];
    }
    pc->sz--;
}

4.查找指定联系人

void SearchContast(contast* pc)
{
    assert(pc);
    if(pc->sz == 0)
    {
       printf("通讯录没有成员\n");
       return;
    }
    char name[NAME_MAX] = 0;
    printf("请输入你要查看的成员姓名:");
    scanf("%s", name);
    int pos = FindByName(pc->data, name);
    if(pos == -1)
    {
       printf("没有该成员,无法查看\n");
       return;    
    }    
    printf("%-20s\t%-5s\t%-4s\t%-20s\t%-12s\n", "姓名","年纪","性别","地址","电话号码");
    printf("%-20s\t%-5d\t%-4s\t%-20s\t%-12s\n",  pc->data[pos].name
                                              ,  pc- >data[pos].age
                                              ,  pc->data[pos].sex
                                              ,  pc->data[pos].addr
                                              ,  pc->data[pos].tele);

}

5.显示所有联系人

void ShowContast(contast* pc)
{
    assert(pc);
    if(pc->sz == 0)
    {
       printf("通讯录没有成员,无法查看\n");
       return;
    }
    int i = 0;
    printf("%-20s\t%-5s\t%-4s\t%-20s\t%-12s\n", "姓名", "年纪", "性别", "地址", "电话号码");
    for(i = 0; i < pc->sz; i++)
    {
        printf("%-20s\t%-5d\t%-4s\t%-20s\t%-12s\n",  pc->data[i].name
                                                  ,  pc->data[i].age
                                                  ,  pc->data[i].gender
                                                  ,  pc->data[i].add
                                                  ,  pc->data[i].tele)
    }
}

6.修改指定联系人

void Modifycontast(contast* pc)
{
    assert(pc);
    if(pc->sz == 0)
    {
       printf("通讯录没有成员,无法进行修改\n");
       return;
    }
    char name[NAME_MAX] = {0};
    printf("请输入修改人的姓名:");
    scanf("%s", name);
    int pos = FindByName(pc->data, name);
    if(pos == -1)
    {
       printf("没有该联系人\n");
       return;
    }
    printf("请输入姓名:>");
    scanf("%s", pc->data[pos].name );
    printf("请输入年纪:>");
    scanf("%d", &(pc->data[pos].age) );
    printf("请输入性别:>");
    scanf("%s", pc->data[pos].sex );
    printf("请输入地址:>");
    scanf("%s", pc->data[pos].addr );
    printf("请输入电话号:>");
    scanf("%s", pc->data[pos].tele );
}    

7.给通讯录排序(根据姓名来排序)

void SortContast(contast* pc)
{
    assert(pc);
    if(pc->sz == 0)
    {
       printf("通讯录没有成员,无法排序\n");
       return;
    }

    qsort( pc->data, 
           pc->sz,
           sizeof(pc->data[0]), 
           cmp_data_PeoInfo_name );

    printf("排列结束\n");
}

int cmp_data_PeoInfo_name(void* e1, void* e2)
{
    return strcmp(((*contast)e1)->name, ((*contast)e2)->name);
}

8.清空所有联系人

void EmptyContast(contast* pc)
{
    assert(pc);
    int i = 0;
    if(pc->sz == 0)
    {
       printf("通讯录为空,没有成员可删除\n");
       return;
    }
    Initcontast(pc);  //  或pc->sz = 0  memset(pc->data, 0, sizeof(pc->data));
    printf("清空完毕\n");  
}

动态版本:

9.通讯录的销毁

void DestoryContast(contast* pc)
{
    free(pc->data);
    pc->data = NULL;
    pc->capacity = 0;
    pc->sz = 0;
    pc = NULL  //将pc置空也可以
}

四.完整代码

创建三个文件,test.c- 测试,contast.h - 函数的声明, contast.c - 函数的定义

test.c

#include "contast.h"

void menu()
{
    printf("*************************");
    printf("****  通讯录管理系统  ****");
    printf("*** 1.Add     2.Del   ***");
    printf("*** 3.Search  4.Modify **");
    printf("*** 5.Show    6.SorT  ***");
    printf("*** 7.empty   0.exit  ***");
    printf("*************************");
}

int main()
{
        int input = 0;
        contast con;
        Initcontast(&con);  //结构体contast初始化
        do
        {
          menu();
          printf("请选择:>");
          scanf("%d", &input);
          switch(input)
          {
            case Add:
                   AddContast(&con);
                   break;    
            case Del: 
                   DelContast(&con);
                   break;  
            case Search:
                   SearchContast(&con);
                   break;  
            case Modify:
                   ModifyContast(&con);
                   break;  
            case Show5:
                   ShowContast(&con);
                   break;  
            case Sort:
                   SortContast(&con);
                   break;  
            case Empty:
                   EmptyContast(&con);
                   break;
            case Exit:
                   printf("退出程序");
                   break;  
            default:
                   DestoryContast(&con);  //动态版本才有
                   printf("没有该指令,请重新输入");
                   break 
          }
        }while(input);

        return 0;
}

contast.h

#include <stdio.h>
#include <string.h>
#include <assert.h>

#define MXA 100
#define NAME_MAX 20
#define ADDR_MAX 30
#define SEX_MAX 5
#define TELE_MAX 20

//动态版本增加的两个宏
#define DEFAULT_SZ 3
#define INC_SZ 2

typedef struct PoeInfo  //通讯录成员的信息
{
    char name[NAME_MAX];
    int age;
    char sex[SEX_MAX];
    char addr[ADDR_MAX];
    char tele[TELE_MAX];
}PeoInfo;

typedef struct contast  //通讯录成员总数
{
    PeoInfo data[MAX];  //通讯录的容量
    int sz;             //现成员的个数
}contast;

enum Function
{
    Exit,
    Add,
    Del,
    Search,
    Modify,
    Show,
    Sort,
    Empty
};         //将功能枚举出来
                 

//增加联系人
void AddContast(contast* pc);

//删除联系人
void DelContast(contast* pc);

//查找指定联系人的信息
void SearchContast(contast* pc);

//修改指定联系人的信息
void ModifyContast(contast* pc);

//显示所有联系人的信息
void ShowContast(contast* pc);

//给通讯录排序
void SortContast(contast* pc);

//清空通讯录
void EmptyContast(contast* pc);

contast.c

#include "contast.h"

//通讯录初始化(静态版本)
void InitContast(contast* pc)
{
    assert(pc);
    pc->sz = 0;
    memset(pc->data, 0, sizeof(pc->data));
}
//通讯录初始化(动态版本)
void InitContast(contast* pc)
{
    assert(pc);
    pc->sz = 0;
    PeoInfo* ptr = (PeoInfo*)calloc(DEFAULT_SZ, sizeof(PeoInfo));
    if(ptr == NULL)
    {
       printf("%s", strerror(errno));
       return;    
    }
    pc->data = ptr;
    pc->capacity = DEFAULT_SZ;
}

//增加联系人(静态版本)
void AddContast(contast* pc)
{
    assert(pc);
    if(pc->sz == MAX)
    { 
       printf("通讯录已满,无法添加\n");
       return;
    }
      printf("请输入姓名:>");
      scanf("%s", pc->data[pc->sz].name );
      printf("请输入年纪:>");
      scanf("%d", &(pc->data[pc->sz].age) );
      printf("请输入性别:>");
      scanf("%s", pc->data[pc->sz].sex );
      printf("请输入地址:>");
      scanf("%s", pc->data[pc->sz].addr );
      printf("请输入电话号:>");
      scanf("%s", pc->data[pc->sz].tele );
      pz->sz++;
}
//增加联系人(动态版本)
//增加容量
void check_capacity(contast* pc)
{
    assert(pc);
    if(pc->sz == pc->capacity)
    {
       PeoInfo* ptr = (*PeoInfo)realloc(pc->data, (capacity + DEFAULT_SZ) + sizeof(PeoInfo));
       if(ptr == NULL)
       {
          perror("cheak_capacity::realloc");
          return;
       }
       pc->data = ptr;
       pc->capacity += 2;
       printf("增容成功\n");
    }
}

void AddContast(contast* pc)
{
    assert(pc);
    check_capacity(pc);
    printf("请输入姓名:>");
    scanf("%s", pc->data[pc->sz].name );
    printf("请输入年纪:>");
    scanf("%d", &(pc->data[pc->sz].age) );
    printf("请输入性别:>");
    scanf("%s", pc->data[pc->sz].sex );
    printf("请输入地址:>");
    scanf("%s", pc->data[pc->sz].addr );
    printf("请输入电话号:>");
    scanf("%s", pc->data[pc->sz].tele );
    pz->sz++;
}

//寻找指定姓名
int FindByName(contast* ps, char* p)
{
    assert(ps && p);
    int i = 0;
    for(; i < ps->sz; i++)
    {
        if(strcmp(ps->data[i].name, p) == 0)
           return i;
    }
    return -1;
}

//删除指定联系人
void DelContast(contast* pc)
{
    assert(pc);
    if(pc->sz == 0)
    {
       printf("没有该成员");
       return;
    }
    char name[NAME_MAX] = {0};
    int del = FindByName(pc->data, name);
    if(del == -1)
    {
       printf("没有该成员");
       return;
    }
    int i = 0;
    for(i = del; i < pc->sz - 1; i++)
    {
        pc->data[i] = pc->data[i + 1];
    }
    pc->sz--;
}

//寻找指定联系人
void SearchContast(contast* pc)
{
    assert(pc);
    if(pc->sz == 0)
    {
       printf("通讯录没有成员");
       return;
    }
    char name[NAME_MAX] = 0;
    printf("请输入你要查看的成员姓名:");
    scanf("%s", name);
    int pos = FindByName(pc->data, name);
    if(pos == -1)
    {
       printf("没有该成员,无法查看");
       return;    
    }    
    printf("%-20s\t%-5s\t%-4s\t%-20s\t%-12s\n", "姓名","年纪","性别","地址","电话号码");
    printf("%-20s\t%-5d\t%-4s\t%-20s\t%-12s\n",  pc->data[pos].name
                                              ,  pc- >data[pos].age
                                              ,  pc->data[pos].sex
                                              ,  pc->data[pos].addr
                                              ,  pc->data[pos].tele);
}

//显示所以联系人
void ShowContast(contast* pc)
{
    assert(pc);
    if(pc->sz == 0)
    {
       printf("通讯录没有成员,无法查看");
       return;
    }
    int i = 0;
    printf("%-20s\t%-5s\t%-4s\t%-20s\t%-12s\n", "姓名", "年纪", "性别", "地址", "电话号码");
    for(i = 0; i < pc->sz; i++)
    {
        printf("%-20s\t%-5d\t%-4s\t%-20s\t%-12s\n",  pc->data[i].name
                                                  ,  pc->data[i].age
                                                  ,  pc->data[i].gender
                                                  ,  pc->data[i].add
                                                  ,  pc->data[i].tele)
    }
}

//修改指定联系人
void Modifycontast(contast* pc)
{
    assert(pc);
    if(pc->sz == 0)
    {
       printf("通讯录没有成员,无法进行修改");
       return;
    }
    char name[NAME_MAX] = {0};
    printf("请输入修改人的姓名:");
    scanf("%s", name);
    int pos = FindByName(pc->data, name);
    if(pos == -1)
    {
       printf("没有该联系人");
       return;
    }
    printf("请输入姓名:>");
    scanf("%s", pc->data[pos].name );
    printf("请输入年纪:>");
    scanf("%d", &(pc->data[pos].age) );
    printf("请输入性别:>");
    scanf("%s", pc->data[pos].sex );
    printf("请输入地址:>");
    scanf("%s", pc->data[pos].addr );
    printf("请输入电话号:>");
    scanf("%s", pc->data[pos].tele );
}   

//给通讯录排序(根据姓名)
int cmp_data_PeoInfo_name(const void* e1, const void* e2)
{
	return strcmp(((PeoInfo*)e1)->name, ((PeoInfo*)e2)->name);
}
void SortContast(contast* pc)
{
	assert(pc);
	if (pc->sz == 0)
	{
		printf("通讯录没有成员,无法排序\n");
		return;
	}
	qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp_data_PeoInfo_name);
	printf("排序完成\n");
	int i = 0;
	printf("%-20s\t%-5s\t%-4s\t%-20s\t%-12s\n", "姓名", "年纪", "性别", "地址", "电话号码");
	for (i = 0; i < pc->sz; i++)
	{
		printf("%-20s\t%-5d\t%-4s\t%-20s\t%-12s\n", pc->data[i].name
			, pc->data[i].age
			, pc->data[i].sex
			, pc->data[i].addr
			, pc->data[i].tele);
	}
}


//清空所有联系人
void EmptyContast(contast* pc)
{
    assert(pc);
    int i = 0;
    if(pc->sz == 0)
    {
       printf("通讯录为空,没有成员可删除");
       return;
    }
    Initcontast(pc);  //  或pc->sz = 0  memset(pc->data, 0, sizeof(pc->data));
    printf("清空完毕\n");  
}

//动态版本
//释放申请的空间
void DestoryContast(contast* pc)
{
    free(pc->data);
    pc->data = NULL;
    pc->capacity = 0;
    pc->sz = 0;
    pc = NULL;
}

以上就是通讯录完整代码,有兴趣的可以观摩一下,它可以加深我们对结构体的掌握

C语言课程设计任务书(4) 一、题目:通讯录管理 二、目的与要求 1. 目的: (1)基本掌握面向过程程序设计的基本思路和方法; (2)达到熟练掌握C语言的基本知识和技能; (3)能够利用所学的基本知识和技能,解决简单的程序设计问题 2. 要求 基本要求: 1.         要求利用C语言面向过程的编程思想来完成系统的设计; 2.       突出C语言的函数特征,以多个函数实现每一个子功能; 3.         画出功能模块图; 4.         具有清晰的程序流程图和数据结构的详细定义; 5.       熟练掌握C语言对文件的各种操作。 创新要求: 在基本要求达到后,可进行创新设计,如系统用户功能控制,对管理员级和一般级别的用户系统功能操作不同 三、信息描述 有关该系统基本信息的描述,如:姓名、电话、城市和邮编等。 四、功能描述 1.       名单基本信息(姓名,城市,电话,邮编等)的录入,并存放在文件当中。 2.       基本信息的查询与修改。 3.       记录的添加和删除。 4.       对同一类型记录的查找:如查找同一城市的记录或同一省份的记录。 五、解决方案 1.       分析程序的功能要求,划分程序功能模块。 2.       画出系统流程图。 3.       代码的编写。定义数据结构和各个功能子函数。 4.       程序的功能调试。 5.       完成系统总结报告以及使用说明书 六、进度安排 此次课程设计时间为一周或两周,分四个阶段完成: 1.       分析设计阶段。指导教师应积极引导学生自主学习和钻研问题,明确设计要求,找出实现方法,按照需求分析、总体设计、详细设计这几个步骤进行。 2.       编码调试阶段:根据设计分析方案编写C代码,然后调试该代码,实现课题要求的功能。 3.       总结报告阶段:总结设计工作,写出课程设计说明书,要求学生写出需求分析、总体设计、详细设计、编码、测试的步骤和内容。 4.       考核阶段。 七、撰写课程设计报告或课程设计总结 课程设计报告要求: 总结报告包括需求分析、总体设计、详细设计、编码(详细写出编程步骤)、测试的步骤和内容、课程设计总结、参考资料等,不符合以上要求者,则本次设计以不及格记。 八、参考资料  《C语言程序设计教程》   网上相关资料(....略)
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值