C语言实现简易通讯录(静态版本)的代码分享

一、通讯录

1.演示效果

2.完整代码

#define _CRT_SECURE_NO_WARNINGS 1
//#include "ConTacts.h"
#include 
#include 
#define MAX 30
#define NAME 20
#define TEL 20
#define ADR 30
struct PopInfo {//声明一个结构体用来保存个人信息
	char name[NAME];//姓名
	int age;//年龄
	char tel[TEL];//手机号
	char adr[ADR];//住址
};
struct Contact {
	struct PopInfo data[MAX];//用结构体数组来存储多个人的信息
	int sz;//用来记录存储几个人的信息
};
void InitContact(struct Contact* con)//初始化联系人信息
{
	assert(con);
	memset(con->data, 0, MAX * sizeof(struct PopInfo));
	con->sz = 0;
}
void AddContact(struct Contact* con)//添加联系人信息
{
	assert(con);
	printf("请输入姓名:");
	scanf("%s", con->data[con->sz].name);
	printf("请输入年龄:");
	scanf("%d", &(con->data[con->sz].age));
	printf("请输入手机号:");
	scanf("%s", con->data[con->sz].tel);
	printf("请输入住址:");
	scanf("%s", con->data[con->sz].adr);
	con->sz++;
	printf("成功添加联系人!\n");
}
void ShowContact(struct Contact* con)//显示所有联系人信息
{
	int i = 0;
	printf("%-20s\t%-20s\t%-20s\t%-20s\t\n", "姓名", "年龄", "手机号", "住址");
	for (i = 0; i < con->sz; i++)
	{
		printf("%-20s\t%-20d\t%-20s\t%-20s\t\n", con->data[i].name, con->data[i].age, con->data[i].tel, con->data[i].adr);
	}
}
int Find(struct Contact* con)//根据姓名查找存放联系人信息的数组下标
{
	char name[20];
	scanf("%s", name);
	int i = 0;
	for (i = 0; i < con->sz; i++)
	{
		if (strcmp(con->data[i].name, name) == 0)
		{
			return i;
			break;
		}
	}
	return -1;
}
void DelContact(struct Contact* con)//删除指定联系人信息函数
{
	assert(con);
	printf("请输入你要删除的联系人姓名:");
	/*char name[20];
	printf("请输入你要删除的联系人姓名:");
	scanf("%s", name);
	int i = 0;
	for (i = 0; i < con->sz; i++)
	{
		if (strcmp(con->data[i].name, name) == 0)
		{
			printf("删除成功!\n");
			con->sz--;
			break;
		}
	}*/
int i = Find(con);
if (i == -1)
{
	printf("未找到你要删除的联系人\n");
}
else
{
	int j = 0;
	for (j = i; j <= con->sz; j++)
	{
		con->data[j] = con->data[j + 1];
	}
	con->sz--;
	printf("删除成功!\n");
}
}
void ModContact(struct Contact* con)//修改联系人信息
{
	assert(con);
	printf("请输入你要修改的联系人姓名:");
	/*char name[20];
	printf("请输入你要修改的联系人姓名:");
	scanf("%s", name);
	int i = 0;
	for (i = 0; i < con->sz; i++)
	{
		if (strcmp(con->data[i].name, name) == 0)
		{
			printf("找到要修改的联系人!\n");
			break;
		}
	}*/
	int i = Find(con);
	if (i == -1)
	{
		printf("未找到你要删除的联系人\n");
	}
	else
	{
		printf("请输入姓名:");
		scanf("%s", con->data[i].name);
		printf("请输入年龄:");
		scanf("%d", &(con->data[i].age));
		printf("请输入手机号:");
		scanf("%s", con->data[i].tel);
		printf("请输入住址:");
		scanf("%s", con->data[i].adr);
		printf("成功修改联系人信息!\n");
	}
}
void SearchContact(struct Contact* con)//查找指定联系人
{
	assert(con);//断言,con不能指向NULL
	printf("请输入你要查找的联系人姓名:");
	/*char name[20];
	printf("请输入你要查找的联系人姓名:");
	scanf("%s", name);
	int i = 0;
	for (i = 0; i < con->sz; i++)
	{
		if (strcmp(con->data[i].name, name) == 0)
		{
			break;
		}
	}*/
	int i = Find(con);
	if (i == -1)
	{
		printf("没有找到你要查找的联系人!\n");
	}
	else
	{
		printf("%-20s\t%-20s\t%-20s\t%-20s\t\n", "姓名", "年龄", "手机号", "住址");
		printf("%-20s\t%-20d\t%-20s\t%-20s\t\n", con->data[i].name, con->data[i].age, con->data[i].tel, con->data[i].adr);
	}
}
int cmp_name(const void* e1, const void* e2)//qsort中的比较函数
{
	return strcmp(((struct PopInfo*)e1)->name, ((struct PopInfo*)e2)->name);//只能对字符及字符串排序,如果是数字,会出错
}
void Sort_name_Contact(struct Contact* con)//qsort函数排序
{
	qsort(con->data, con->sz, sizeof(struct PopInfo), cmp_name);
	printf("排序成功\n");//名字按照字母的ASCII码值从小到大排序
}
void ClearContact(struct Contact* con)//清空所有联系人信息
{
	char str[10];
	while (1)
	{
		printf("确定清空通讯录吗?(yes/no):");
		scanf("%s", str);
		if (strcmp(str, "yes") == 0)
		{
			con->sz = 0;
			printf("已清空通讯录!\n");
			break;
		}
		else if (strcmp(str, "no") == 0)
		{
			printf("已取消!\n");
			break;
		}
		else
			printf("输入错误,请重新输入!\n");
	}
}
enum um {//用枚举类型表示input的值
	Exit,//Exit是常量值0,退出程序
	Add,//Add是常量值1,添加联系人信息
	Del,//Del是常量值2,删除指定联系人信息
	Mod,//3,修改联系人信息
	Search,//4,查找联系人信息
	Sort_name,//5,按照姓氏对联系人信息排序
	Show,//6,显示联系人信息
	Clear//7,清空所有联系人信息
};

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");
}
int main()
{
	int input = 0;
	struct Contact con;
	InitContact(&con);//初始化联系人信息
	do {
		menu();
		printf("请选择:");
		scanf("%d", &input);
		switch (input)
		{
		case Add:
			AddContact(&con);//添加联系人信息
			break;//break仅退出switch语句,不会退出do-while循环
		case Del:
			DelContact(&con);//删除指定联系人信息
			break;
		case Mod:
			ModContact(&con);//修改联系人信息
			break;
		case Search:
			SearchContact(&con);//查找指定联系人
			break;
		case Sort_name:
			Sort_name_Contact(&con);//按照姓氏对联系人信息排序
			break;
		case Show:
			ShowContact(&con);//显示所有联系人信息
			break;
		case Clear:
			ClearContact(&con);//清空所有联系人信息
			break;
		case Exit:
			printf("退出程序\n");
			break;
		default:
			printf("输入错误,请重新选择\n");
			break;
		}
	} while (input);
	return 0;
}

二、代码解析

1.宏定义及结构体声明

#include 
#include 
#define MAX 30
#define NAME 20
#define TEL 20
#define ADR 30
struct PopInfo {//声明一个结构体用来保存个人信息
    char name[NAME];//姓名
    int age;//年龄
    char tel[TEL];//手机号
    char adr[ADR];//住址
};
struct Contact {
    struct PopInfo data[MAX];//用结构体数组来存储多个人的信息
    int sz;//用来记录存储几个人的信息
};

定义struct PopInfo结构体用来保存联系人的信息,再定义struct Contact,用来保存多个联系人的信息以及记录所存储联系人个数。

2.主菜单函数

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");
}

3.主函数

int main()
{
    enum um {//用枚举类型表示input的值
        Exit,//Exit是常量值0,退出程序
        Add,//Add是常量值1,添加联系人信息
        Del,//Del是常量值2,删除指定联系人信息
        Mod,//3,修改联系人信息
        Search,//4,查找联系人信息
        Sort_name,//5,按照姓氏对联系人信息排序
        Show,//6,显示联系人信息
        Clear//7,清空所有联系人信息
    };
    int input = 0;
    struct Contact con;
    InitContact(&con);//初始化联系人信息
    do {
        menu();
        printf("请选择:");
        scanf("%d", &input);
        switch (input)
        {
        case Add:
            AddContact(&con);//添加联系人信息
            break;//break仅退出switch语句,不会退出do-while循环
        case Del:
            DelContact(&con);//删除指定联系人信息函数
            break;
        case Mod:
            ModContact(&con);//修改联系人信息
            break;
        case Search:
            SearchContact(&con);//查找指定联系人
            break;
        case Sort_name:
            Sort_name_Contact(&con);//按照姓氏对联系人信息排序
            break;
        case Show:
            ShowContact(&con);//显示所有联系人信息
            break;
        case Clear:
            ClearContact(&con);
            break;
        case Exit:
            printf("退出程序\n");
            break;
        default:
            printf("输入错误,请重新选择\n");
            break;
        }
    } while (input);
    return 0;
}

4.查找函数

int Find(struct Contact* con)//根据姓名查找存放联系人信息的数组下标
{
    char name[20];
    scanf("%s", name);
    int i = 0;
    for (i = 0; i < con->sz; i++)
    {
        if (strcmp(con->data[i].name, name) == 0)
        {
            return i;
            break;
        }
    }
    return -1;
}

根据姓名查找存放联系人信息的数组下标,用于后来查找、删除、修改联系人信息时使用。

5.初始化联系人信息

void InitContact(struct Contact* con)//初始化联系人信息
{
    assert(con);
    memset(con->data, 0, MAX * sizeof(struct PopInfo));
    con->sz = 0;
}

assert(con);是判断指针con是否为空,判断memset(con->data, 0, MAX * sizeof(struct PopInfo))是将结构体struct Contact中struct PopInfo类型的数组data置为0。con->sz = 0;是将所记录的联系人个数置为0。

6.添加联系人信息

void AddContact(struct Contact* con)//添加联系人信息
{
    assert(con);
    printf("请输入姓名:");
    scanf("%s", con->data[con->sz].name);
    printf("请输入年龄:");
    scanf("%d", &(con->data[con->sz].age));
    printf("请输入手机号:");
    scanf("%s", con->data[con->sz].tel);
    printf("请输入住址:");
    scanf("%s", con->data[con->sz].adr);
    con->sz++;
    printf("成功添加联系人!\n");
}

注意在输入年龄的时候因为年龄是int类型,要用&符号取出地址,再进行赋值。

7.显示所有联系人信息

void ShowContact(struct Contact* con)//显示所有联系人信息
{
    int i = 0;
    printf("%-20s\t%-20s\t%-20s\t%-20s\t\n", "姓名", "年龄", "手机号", "住址");
    for (i = 0; i < con->sz; i++)
    {
        printf("%-20s\t%-20d\t%-20s\t%-20s\t\n", con->data[i].name, con->data[i].age, con->data[i].tel, con->data[i].adr);
    }
}

printf("%-20s\t%-20s\t%-20s\t%-20s\t\n", "姓名", "年龄", "手机号", "住址")中的%-20s\t%-20s\t%-20s\t%-20s\t\n是为了让打印联系人信息的时候更加直观,%-20s\t%-20s\t%-20s\t%-20s\t\n中的-是为了让其左对齐,把-去掉可以是右对齐。

8.删除指定联系人信息

void DelContact(struct Contact* con)//删除指定联系人信息函数
{
    assert(con);
    printf("请输入你要删除的联系人姓名:");
    /*char name[20];
    printf("请输入你要删除的联系人姓名:");
    scanf("%s", name);
    int i = 0;
    for (i = 0; i < con->sz; i++)
    {
        if (strcmp(con->data[i].name, name) == 0)
        {
            printf("删除成功!\n");
            con->sz--;
            break;
        }
    }*/
int i = Find(con);
if (i == -1)
{
    printf("未找到你要删除的联系人\n");
}
else
{
    int j = 0;
    for (j = i; j <= con->sz; j++)
    {
        con->data[j] = con->data[j + 1];
    }
    con->sz--;
    printf("删除成功!\n");
}
}

代码中注释部分可以在不调用 Find函数时删除联系人信息,在成功删除联系人信息后,要将存放联系人个数的变量进行修改。

9.修改联系人信息

void ModContact(struct Contact* con)//修改联系人信息
{
    assert(con);
    printf("请输入你要修改的联系人姓名:");
    /*char name[20];
    printf("请输入你要修改的联系人姓名:");
    scanf("%s", name);
    int i = 0;
    for (i = 0; i < con->sz; i++)
    {
        if (strcmp(con->data[i].name, name) == 0)
        {
            printf("找到要修改的联系人!\n");
            break;
        }
    }*/
    int i = Find(con);
    if (i == -1)
    {
        printf("未找到你要删除的联系人\n");
    }
    else
    {
        printf("请输入姓名:");
        scanf("%s", con->data[i].name);
        printf("请输入年龄:");
        scanf("%d", &(con->data[i].age));
        printf("请输入手机号:");
        scanf("%s", con->data[i].tel);
        printf("请输入住址:");
        scanf("%s", con->data[i].adr);
        printf("成功修改联系人信息!\n");
    }
}

10.查找指定联系人

void SearchContact(struct Contact* con)//查找指定联系人
{
    assert(con);//断言,con不能指向NULL
    printf("请输入你要查找的联系人姓名:");
    /*char name[20];
    printf("请输入你要查找的联系人姓名:");
    scanf("%s", name);
    int i = 0;
    for (i = 0; i < con->sz; i++)
    {
        if (strcmp(con->data[i].name, name) == 0)
        {
            break;
        }
    }*/
    int i = Find(con);
    if (i == -1)
    {
        printf("没有找到你要查找的联系人!\n");
    }
    else
    {
        printf("%-20s\t%-20s\t%-20s\t%-20s\t\n", "姓名", "年龄", "手机号", "住址");
        printf("%-20s\t%-20d\t%-20s\t%-20s\t\n", con->data[i].name, con->data[i].age, con->data[i].tel, con->data[i].adr);
    }
}

11.按照姓氏对联系人信息排序

int cmp_name(const void* e1, const void* e2)//qsort中的比较函数
{
    return strcmp(((struct PopInfo*)e1)->name, ((struct PopInfo*)e2)->name);//只能对字符及字符串排序,如果是数字,会出错
}
void Sort_name_Contact(struct Contact* con)//qsort函数排序
{
    qsort(con->data, con->sz, sizeof(struct PopInfo), cmp_name);
    printf("排序成功\n");//名字按照字母的ASCII码值从小到大排序
}

使用 qsort 函数对其进行排序。

12.清空所有联系人信息

void ClearContact(struct Contact* con)//清空所有联系人信息
{
	char str[10];
	while (1)
	{
		printf("确定清空通讯录吗?(yes/no):");
		scanf("%s", str);
		if (strcmp(str, "yes") == 0)
		{
			con->sz = 0;
			printf("已清空通讯录!\n");
			break;
		}
		else if (strcmp(str, "no") == 0)
		{
			printf("已取消!\n");
			break;
		}
		else
			printf("输入错误,请重新输入!\n");
	}
}

到此这篇关于C语言实现简易通讯录(静态版本)的代码分享的文章就介绍到这了,更多相关C语言静态通讯录内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

你可能感兴趣的:(C语言实现简易通讯录(静态版本)的代码分享)