【C语言】通讯录管理系统(容量可变)

在实现容量可变的通讯录之前,我们得想弄清楚一下几个函数。
malloc函数
函数原型:
extern void *malloc(unsigned int num_bytes);
功能: 分配长度为num_bytes字节的内存块
返回值: 返回值类型是void *,所以我们可以把分配好的空间强制类型转化成你要使用的类型。如果内存分配成功则返回指向被分配内存的指针,否则返回空指针NULL。当内存不再使用时,应使用free()函数将内存块释放。函数返回的指针一定要适当对齐,使其可以用于任何数据对象。

calloc函数
函数原型:
void* calloc (size_t num, size_t size);
功能: 用来动态地分配内存空间并初始化为0,calloc() 在内存中动态地分配 num 个长度为size的连续空间,并将每一个字节都初始化为0.所以它的结果是分配了 num*size 个字节长度的内存空间,并且每个字节的值都是0。
返回值 : 返回值类型是void *,所以我们可以把分配好的空间强制类型转化成你要使用的类型。分配成功返回指向该内存的地址,失败则返回 NULL。

注意:如果 size 的值为 0,那么返回值会因标准库实现的不同而不同,可能是 NULL,也可能不是,但返回的指针不应该再次被引用。

realloc函数
函数原型:
extern void *realloc(void *mem_address, unsigned int newsize);
功能: 更改已经分配的内存空间,更改的大小可大可小。
1. 如果新的大小小于原内存大小,realloc仅仅是改变索引的信息,但这样可能会导致数据丢失,慎用!

2 . 如果新的大小大于原内存大小,则有以下情况:
先判断当前的指针是否有足够的连续空间。
1)如果当前内存段后面有足够的内存空间,则直接扩展这段内存空间,即扩大mem_address指向的地址,并且将mem_address返回。
2)如果当前内存段后面的内存空间不够分配,则先按照newsize指定的大小重新分配一段空间,将原空间的数据从头到尾拷贝到新分配的内存空间,而后释放原来mem_address所指内存空间(注意:原指针是自动释放,不需要使用free),同时返回新分配的内存空间的首地址。即重新分配存储器块的地址。
3)如果申请失败,将返回NULL,此时,原来的指针仍然有效。

注意:如果调用成功,不管当前内存段后面的空闲空间是否满足要求,都会释放掉原来的指针,重新返回一个指针,只是返回的指针有可能和原来的指针一样,这也意味着不能再次释放掉原来的指针。

以上就是关于这几个函数的介绍。因为经常会用到这几个函数,所以一定要掌握。这次的容量可变的通讯录正是使用了这几个函数,来实现动态增容的效果。

tongxunlu.h

#ifndef _TONGXUNLU_H_
#define _TONGXUNLU_H_
#define _CRT_SECURE_NO_WARNINGS

#define NAME_MAXNUM 20
#define SEX_MAXNUM 5
#define TELE_MAXNUM 12
#define ADDR_MAXNUM 10
#define MAX 1000
#define MAX_INIT 2   //最大容量
#define MAX_RISE 2   //增加容量
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<assert.h>


typedef struct person 
{
    char name[NAME_MAXNUM];
    int age;
    char sex[SEX_MAXNUM];
    char tele[TELE_MAXNUM];
    char addr[ADDR_MAXNUM];
}person;

typedef struct tongxunlu_list
{
    person *data;       //指向联系人的指针
    int size;            //联系人的个数
    int capacity;        //通讯录的容量
}list,*list_p;


void Add(list_p List);        //添加联系人
void Display(list_p List);    //显示联系人信息
void Del(list_p List);        //删除指定联系人
void Search(list_p List);     //查找指定联系人
void Revise(list_p List);     //修改指定联系人
void Sort_name(list_p List);  //联系人按照姓名排序
void Empty(list_p List);      //清空联系人


#endif

tongxunlu.c

#include"tongxunlu.h"


static void init(list_p List)   //初始化通讯录空间
{
    List->data=(person *)malloc(MAX_INIT*sizeof(person));
    if(List->data == NULL)
    {
        assert(false);
    }
    List->size=0;
    List->capacity=MAX_INIT;
}

void menu()
{
    printf("***************简易通讯录***************\n");
    printf("****1.添加联系人*******2.删除联系人****\n");
    printf("****3.查找联系人*******4.修改联系人****\n");
    printf("****5.显示联系人*******6.清空联系人****\n");
    printf("****7.以名字排序联系人*0.退出**********\n");
    printf("***************************************\n");
}

void Add(list_p List)      //添加联系人
{
    if(List->size >= List->capacity)
    {
        person *temp=(person *)realloc(List->data,(List->capacity+MAX_RISE)*sizeof(person));  
                    //当实际联系人个数和初始化的容量相等时用realloc增容
        if(temp == NULL)
        {
            assert(false);
        }
        else
        {
            List->data=temp;
            List->capacity+=MAX_RISE;
        }
    }
    printf("请输入你添加之后的联系人信息:(姓名 年龄 性别 联系电话 住址):\n");
    scanf("%s",List->data[List->size].name);
    scanf("%d",&(List->data[List->size].age));
    scanf("%s",List->data[List->size].sex);
    scanf("%s",List->data[List->size].tele);
    scanf("%s",List->data[List->size].addr);
    List->size++;
    printf("添加成功\n");
}

static int Find_position(list_p List,char *str)   //查找
{
    int i=0;
    for(i=0;i<List->size;i++)
    {
        if(strcmp(List->data[i].name,str) == 0)  //将要查找的姓名与通讯录里的联系人匹配,找到就返回下标
        {
            return i;
        }
    }
    return -1;
}

void Del(list_p List)  //删除指定联系人
{
    int ret=0;
    int i=0;
    char str[NAME_MAXNUM]={0};
    printf("请输入你要删除的联系人姓名:");
    scanf("%s",str);
    ret=Find_position(List,str);
    if(ret != -1)
    {
        for(i=ret;i<List->size;i++)
        {
            List->data[i]=List->data[i+1];  //删除的元素之后的元素往前挪一个位置
        }
        List->size--;
    }
    else
    {
        printf("通讯录里没有此联系人\n");
        return ;
    }
}

void Search(list_p List)  //查找指定联系人
{

    int ret=0;
    char str[NAME_MAXNUM]={0};
    printf("请输入你要查找的联系人的名字:");
    scanf("%s",str);
    ret=Find_position(List,str);
    if(ret != -1)
    {
        printf("查找成功\n");
    }
    else
    {
        printf("通讯录里无此人\n");
        return ;
    }
}

void Revise(list_p List)  //修改指定联系人
{
    int ret=0;
    char str[NAME_MAXNUM]={0};
    printf("请输入你要修改的联系人的姓名:");
    scanf("%s",str);
    ret=Find_position(List,str);
    if(ret != -1)
    {
        int input=0;
        printf("请输入你要修改的信息:1.姓名2.年龄3.性别4.联系电话5.住址\n");
        scanf("%d",&input);
        switch(input)
        {
        case 1:printf("请输入修改后的名字:\n");
              scanf("%s",List->data[ret].name);break;
        case 2:printf("请输入修改后的年龄:\n");
              scanf("%d",&(List->data[ret].age));break;
        case 3:printf("请输入修改后的性别:\n");
              scanf("%s",List->data[ret].sex);break;
        case 4:printf("请输入修改后的联系电话:\n");
              scanf("%s",List->data[ret].tele);break;
        case 5:printf("请输入修改后的住址:\n");
              scanf("%s",List->data[ret].addr);break;
        default:printf("error");
            break;
        }
    }
    else
    {
        printf("通讯录里无此人\n");
        return ;
    }
}

void Display(list_p List)  //显示联系人信息
{
    int i=0;
    for(i=0;i<List->size;i++)
    {
        printf("%s %d %s %s %s\n",List->data[i].name,List->data[i].age,List->data[i].sex,\
                                  List->data[i].tele,List->data[i].addr);
    }
}

void Empty(list_p List)   //清空联系人列表
{
    List->size=0;
}

void Sort_name(list_p List)  //以名字排序联系人
{
    int flag=0;
    int i=0;
    int j=0;
    for(i=0;i<List->size-1;i++)
    {
        flag=0;   //对冒泡排序的优化
        for(j=0;j<List->size-1-i;j++)
        {
            if(strcmp(List->data[j].name,List->data[j+1].name) > 0) //默认升序排列
            {
                person temp=List->data[j];
                List->data[j]=List->data[j+1];
                List->data[j+1]=temp;
                flag=1;
            }
        }
        if(flag == 0)
            break;
    }
}

static void Free(list_p List)   //释放通讯录空间
{
    free(List->data);
    List->data=NULL;
    exit(0);
}

int main()
{
    list List;
    int intput=1;
    init(&List);
    while(intput)
    {
        menu();
        printf("请输入一个你要选择的操作:");
        scanf("%d",&intput);
        if(intput >= 0 && intput <= 7)
        {
                switch(intput)
                {
                case 1:Add(&List);
                    break;
                case 2:Del(&List);
                    break;
                case 3:Search(&List);
                    break;
                case 4:Revise(&List);
                    break;
                case 5:Display(&List);
                    break;
                case 6:Empty(&List);
                    break;
                case 7:Sort_name(&List);
                    break;
                case 0:Free(&List);
                    break;
                }
        }
    }
    system("pause");
    return 0;
}
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语言程序设计教程》   网上相关资料(....略)
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值