通讯录实现
Suzhou 50 1

contact.h:

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

#define MAX 1000

#define MAX_NAME 10
#define MAX_SEX 3
#define MAX_TEL 12
#define MAX_ADDR 20

//使用枚举增加代码可读性
enum Option
{
    EXIT,  //0
    ADD,  //1
    DEL,  //2
    SEARCH,  //3
    MODIFY,  //4
    SHOW,  //5
};

struct PeoInfo  //描述一个人的信息
{
    char name[MAX_NAME];
    int age;
    char sex[MAX_SEX];
    char tel[MAX_TEL];
    char addr[MAX_ADDR];
};


struct Contact  //描述通讯录
{
    struct PeoInfo data[MAX];  //存放1000个人的信息
    int size; //记录当前已有的元素个数
};


//声明函数:
void InitContact(struct Contact* ps);  //初始化函数,参数为struct Contact的指针    
void AddContact(struct Contact* ps);   //数组内增加元素
void ShowContact(const struct Contact* ps);  //打印数组
void DelContact(struct Contact* ps);  //删除指定信息
void SearContact(const struct Contact* ps);  //查找指定信息
void ModContact(struct Contact* ps);  //更改指定信息

contact.c:

#define _CRT_SECURE_NO_WARNINGS

#include "contact.h"

void InitContact(struct Contact* ps)
{
    memset(ps->data, 0, sizeof(ps->data));
    //memset将ps->data的内存空间设置成0,需要设置为0的字节数是sizeof(ps->data)。
    //data是struct PeoInfo类型的数组,ps->data找到这个数组,数组名在sizeof内部表示整个数组
    ps->size = 0;  //通讯录最初是空白的
}

void AddContact(struct Contact* ps)
{
    //先检测数组中是否有空间可供新元素存储
    if (ps->size == MAX)
    {
        printf("通讯录已满,无法增加\n");
    }
    else
    {
        //当数组中已经有元素了,新增的元素需要存放在后面,设已有5个元素,size=5,占据的数组元素下标为
        //0、1、2、3、4,新增元素可以存放在下表为5的位置,ps->data找到该数组,ps->size该元素下标,
        //.操作符表示存放在该结构体数组元素的成员名
        printf("请输入姓名\n");
        scanf("%s", &ps->data[ps->size].name);
        printf("请输入年龄\n");
        scanf("%d", &ps->data[ps->size].age);
        printf("请输入性别\n");
        scanf("%s", &ps->data[ps->size].sex);
        printf("请输入电话\n");
        scanf("%s", &ps->data[ps->size].tel);
        printf("请输入地址\n");
        scanf("%s", &ps->data[ps->size].addr);

        ps->size++;  //数组中成员数量+1
        printf("添加成功\n");
    }
}

void ShowContact(const struct Contact* ps)
{
    if (ps->size == 0)
    {
        printf("通讯录为空\n");
    }
    else
    {
        int i = 0;
        printf("%-10s\t%-3s\t%-5s\t%-12s\t%-20s\n", "姓名", "性别", "年龄", "电话", "地址");
        for (i = 0; i < ps->size; i++)
        {
            printf("%-10s\t%-3s\t%-5d\t%-12s\t%-20s\n",
                ps->data[i].name,
                ps->data[i].sex,
                ps->data[i].age,
                ps->data[i].tel,
                ps->data[i].addr);
        }
    }
}

//删除、查找、改变都需要查找代码,代码冗余,单独写成函数调用
int FindName(const struct Contact* ps, char name[MAX_NAME]) 
{
    //跳出for循环的两种情况:
    //1.break跳出,说明数组中找到了要删除的数据
    //2.for循环跳出,说明遍历完成后没有找到查找的数据
    int i = 0;
    for (i = 0; i < ps->size; i++)
    {
        if (strcmp(ps->data[i].name, name) == 0)
        {
            return i;  //找到了返回元素下标
        }
    }
    return -1;  //找不到返回-1
}

void DelContact(struct Contact* ps)
{
    char name[MAX_NAME];
    printf("请输入要删除的人的名字:->");
    scanf("%s", &name);
    //1.查找要删除的信息在数组中的位置(遍历)
    int pos = FindName(ps, name);  //返回位置或-1
    if (pos == -1)  //for循环跳出
    {
        printf("要删除的人不存在\n");
    }
    else  //break跳出,此时的位置是data[i]
    {
    //2.删除信息(保证顺序不变,后面的信息逐个覆盖前面的)
        int j = 0;
        for (j = pos; j < ps->size - 1; j++)
        {
            ps->data[j] = ps->data[j + 1];
        }
        ps->size--;
        printf("删除成功\n");
    }

}

void SearContact(const struct Contact* ps)
{
    char name[MAX_NAME];
    printf("请输入要查找的联系人的名字\n");
    scanf("%s", &name);
    int pos = FindName(ps, name);  //返回位置或-1
    if (pos == -1)
    {
        printf("要查找的人不存在\n");
    }
    else  //找到后打印元素的每一个字段
    {
        printf("%-10s\t%-3s\t%-5s\t%-12s\t%-20s\n", 
            "姓名", "性别", "年龄", "电话", "地址");
        printf("%-10s\t%-3s\t%-5d\t%-12s\t%-20s\n",
            ps->data[pos].name,
            ps->data[pos].sex,
            ps->data[pos].age,
            ps->data[pos].tel,
            ps->data[pos].addr);
    }
}

void ModContact(struct Contact* ps)
{
    char name[MAX_NAME];
    printf("请输入要修改的联系人的名字\n");
    scanf("%s", &name);
    int pos = FindName(ps, name);  //返回位置或-1
    if (pos == -1)
    {
        printf("要修改的信息不存在\n");
    }
    else  //找到后打印元素的每一个字段
    {
        printf("请输入姓名\n");
        scanf("%s", &ps->data[pos].name);
        printf("请输入年龄\n");
        scanf("%d", &ps->data[pos].age);
        printf("请输入性别\n");
        scanf("%s", &ps->data[pos].sex);
        printf("请输入电话\n");
        scanf("%s", &ps->data[pos].tel);
        printf("请输入地址\n");
        scanf("%s", &ps->data[pos].addr);
        printf("修改完成\n");
    }
}

test.c:

#define _CRT_SECURE_NO_WARNINGS
#include "contact.h"

void menu()
{
    printf("****************-***************\n");
    printf("****1.add          2.del    ****\n");
    printf("****3.search       4.modify ****\n");
    printf("****5.show         0.exit   ****\n");
    printf("********************************\n");
}

int main()
{
    int input = 1;
    struct Contact con;  //通讯录con中包含当前数组中的元素个数size和数组容量MAX
    InitContact(&con);  //初始化通讯录,需要把con和size置为0,传址调用
    do
    {
        menu();
        printf("请选择->");
        scanf("%d", &input);
        switch (input)
        {
        //预编译时可以将ADD等改成0...
        case ADD:
            AddContact(&con);  //元素增加后size随之改变,传址调用
            break;
        case DEL:
            DelContact(&con);
            break;
        case SEARCH:
            SearContact(&con);
            break;
        case MODIFY:
            ModContact(&con);
            break;
        case SHOW:
            ShowContact(&con);
            break;
        case EXIT:
            printf("退出通讯录\n");
            break;
        default:
            printf("选择错误\n");
            break;
        }
    } while (input);
    return 0;
}

::: tip 代码文件contact.c中的FindNmae函数只在contact.c中出现,并没有在头文件中声明,且在前FindName函数前加了static,原因是FindName函数不是通讯录的必要功能,只是为了支撑功能实现而写的函数,用户无需知道它的存在,所以不需要暴露接口,在文件中定义后就可以使用。static的作用是使得FindName函数只在源文件contact.c内部可以看到和使用。需要暴露在外面的只是头文件中声明的功能函数。 :::

评论区

索引目录