目录

前言

1、通讯录框架,基本介绍

2、程序实现,步骤+解释

2.1. 建立菜单

2.2. 主函数框架

2.3 前期工作

2.4 初始化通讯录函数:void InitContact(Contact* pc);

2.5 添加信息操作:void AddContact(Contact* pc);

2.6 显示通讯录信息:void ShowContact(Contact* pc);

2.7 删除通讯录信息:void DelContact(Contact* pc);

2.8 查找通讯录信息:void SearchContact(Contact* pc);

2.9 修改通讯录信息:void ModifyContact(Contact* pc);

2.10 排序通讯录信息:void SortContact(Contact* pc);

3 、效果展示

总结


前言

        此篇介绍了一个C语言的项目:通讯录。该项目是C语言进阶的实战项目,希望对你有帮助。

1、通讯录框架,基本介绍

        一个通讯录的人包含的信息:名字;年龄;性别;电话;地址。

        通讯录基本要求:

        1、能存放100人的信息

        2、能增加联系人

        3、能删除指定联系人

        4、查找联系人

        5、修改联系人

        6、排序

        7、显示联系人

2、程序实现,步骤+解释

2.1. 建立菜单

        首先我们运行程序时必须有一个菜单,指引我们如何操作:C语言:通讯录-LMLPHP

        如上图。这样的菜单建立很简单,只需要用到printf函数打印即可:

    printf("*************************************\n");
	printf("******  1. add      2. del     ******\n");
	printf("******  3. search   4. modify  ******\n");
	printf("******  5. show     6. sort    ******\n");
	printf("******  0. exit                ******\n");
	printf("*************************************\n");

        我们将其封装成一个函数,以便我们多次调用。

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

2.2. 主函数框架

int main()
{
	int input = 0;
	Contact con;		//通讯录
	//初始化通讯录
	InitContact(&con);
	do
	{
		menu();
		printf("请选择:>");
		scanf("%d", &input);
		switch (input)
		{
		case 1:		//增加
			AddContact(&con);
			break;
		case 2:		//删除
			DelContact(&con);
			break;
		case 3:		//查找
			SearchContact(&con);
			break;	
		case 4:		//修改
			ModifyContact(&con);
			break;
		case 5:		//显示
			ShowContact(&con);
			break;
		case 6:		//排序
			SortContact(&con);
			break;
		case 0:
			printf("退出通讯录\n");
			break;
		default:
			printf("选择错误,请重新输入\n");
			break;
		}
	} while(input);

	return 0;
}

        主函数的框架其实很简单,一开始时初始化一个通讯录,用到我们自己定义的函数:InitContact()。然后读取一个输入值,这个值是用户输入的,用户根据菜单打印他想要进行的操作,包括增删查改、排序和显示。此内容于菜单对应。输入1:增加信息;输入2:删除信息;输入3:查找信息;输入4:修改信息;输入5:显示信息;输入6:对信息排序;输入0:退出;其他输入:输入错误,重新输入。这样的逻辑我们用一个switch语句恰好实现,如上图。接下来我们要做的就是实现我们主函数中出现的函数的操作了。

2.3 前期工作

        定义通讯录结构体:

typedef struct Contact
{
	PeoInfo data[100];//存放人的信息
	int count;		//记录当前通讯录中实际存储人的信息个数。
}Contact;

        此结构体包括,一个存放人的信息的结构体,和一个记录通讯录存储信息个数的变量。

        PeoInfo结构体定义:

typedef struct PeoInfo
{
	char name[20];
	int age;
	char sex[10];
	char tele[12];
	char addr[30];
}PeoInfo;

        存放人信息的结构体PeoInfo内容包括:姓名;年龄;性别;电话;地址。

2.4 初始化通讯录函数:void InitContact(Contact* pc);

void InitContact(Contact* pc)
{
	assert(pc);
	pc->count = 0;
	memset(pc->data, 0, sizeof(pc->data));
}

        利用memset函数,将传输过来的通讯录结构体初始化,前提包含头文件:string.h。

2.5 添加信息操作:void AddContact(Contact* pc);

void AddContact(Contact* pc)
{
	assert(pc);
	if (pc->count == MAX)
	{
		printf("通讯录已满,无法添加\n");
		return;
	}
	//
	printf("请输入名字:>");
	scanf("%s", pc->data[pc->count].name);
	printf("请输入年龄:>");
	scanf("%d", &pc->data[pc->count].age);
	printf("请输入性别:>");
	scanf("%s", pc->data[pc->count].sex);
	printf("请输入电话:>");
	scanf("%s", pc->data[pc->count].tele);
	printf("请输入地址:>");
	scanf("%s", pc->data[pc->count].addr);
	pc->count++;
	printf("增加成功\n");
}

        其中MAX表示通讯录最大容量,目前设置为100。当通讯录已满时,无法加入新信息。

        通讯录未满,则要求用户对每一个量进行输入,程序将其放在对应位置,最后count++,记录通讯录信息个数的变量加一,表示通讯录又多加了一条信息。

        看效果:        C语言:通讯录-LMLPHP

        可以看到,成功添加了该信息,只是我们测试时还用到了“5”的功能,“5”的功能是显示通讯录现有信息,后面会讲解如何实现。

2.6 显示通讯录信息:void ShowContact(Contact* pc);

        我们不按照数字顺序来,我们按照自己的逻辑顺序进行讲解,我先讲这个“5”的功能是怎么实现的,也就是如何显示通讯录:

void ShowContact(Contact* pc)
{
	assert(pc);
	printf("%-20s\t%-5s\t%-5s\t%-12s\t%-30s\n", "姓名", "年龄", "性别", "电话", "地址");
	int i = 0;
	for (i = 0;i < pc->count;i++)
	{
		printf("%-20s\t", pc->data[i].name);
		printf("%-5d\t", pc->data[i].age);
		printf("%-5s\t", pc->data[i].sex);
		printf("%-12s\t", pc->data[i].tele);
		printf("%-30s\n", pc->data[i].addr);
	}
}

        其实逻辑很简单,就是从0开始,一直到count,也就是通讯录有的信息个数,每一个信息都全部打印一遍,用一个for循环就好了。效果就不展示了,上面已经展示了。

2.7 删除通讯录信息:void DelContact(Contact* pc);

void DelContact(Contact* pc)
{
	assert(pc);
	printf("请输入要删除的人名:>");
	char name[20] = { 0 };
	scanf("%s", name);
	int flag = FindName(pc, name);
	if (flag == -1)
	{
		printf("要删除的信息不存在\n");
		return;
	}
	else
	{
		int i = 0;
		for (i = flag;i < pc->count - 1;i++)
		{
			pc->data[i] = pc->data[i + 1];
		}
		pc->count--;
		printf("删除成功。\n");
	}

}

         删除信息的逻辑是:先通过用户输入的名字找到该信息所在的位置,然后用后面的信息覆盖掉这个信息:后面所有信息都向前挪1位。count自减1,表示通讯录存储信息少一个。

        这中间需要用到查找信息的函数:FindName。我们来看看它如何实现:

        FindName输入参数为:通讯录结构体指针;要查找的名字字符串

        返回值为:如果找到返回该数据在data中的位置,找不到返回-1。

static int FindName(Contact* pc,char name[])
{
	assert(pc);
	int i = 0;
	for (i = 0;i < pc->count;i++)
	{
		if (0 == strcmp(pc->data[i].name, name))
		{
			return i;
		}
	}

	return -1;
}

        static是为了保密,这个作用就相当于给函数加锁,防止给别人看到。可以不加,函数照样实现。

        此函数逻辑为,遍历整个通讯录,找到名字和输入的一致时,返回该下表,遍历到通讯录所记录的信息个数时还未找到,返回-1。

        删除通讯录信息,效果:

2.8 查找通讯录信息:void SearchContact(Contact* pc);

void SearchContact(Contact* pc)
{
	assert(pc);
	printf("请输入要查找人的名字:");
	char name[20] = { 0 };
	scanf("%s", name);
	//查找
	int flag = FindName(pc, name);
	if (flag == -1)
	{
		printf("找不到该信息。\n");
		return;
	}
	else
	{
		printf("%-20s\t%-5d\t%-5s\t%-12s\t%-30s\n",pc->data[flag].name,pc->data[flag].age, pc->data[flag].sex, pc->data[flag].tele, pc->data[flag].addr);
	}
}

        根据输入名字查找,再次运用到函数:FindName。将其返回下表对应的数据输出,如果返回-1,则打印:找不到该信息。

2.9 修改通讯录信息:void ModifyContact(Contact* pc);

void ModifyContact(Contact* pc)
{
	assert(pc);
	printf("请输入要修改的人名:");
	char name[20] = { 0 };
	scanf("%s", name);
	//查找
	int flag = FindName(pc, name);
	if (flag == -1)
	{
		printf("找不到该信息。\n");
		return;
	}
	else
	{
		printf("请输入修改后的名字:>");
		scanf("%s", pc->data[flag].name);
		printf("请输入修改后的年龄:>");
		scanf("%d", &pc->data[flag].age);
		printf("请输入修改后的性别:>");
		scanf("%s", pc->data[flag].sex);
		printf("请输入修改后的电话:>");
		scanf("%s", pc->data[flag].tele);
		printf("请输入修改后的地址:>");
		scanf("%s", pc->data[flag].addr);
		printf("修改成功\n");
	}
}

        仍旧是先通过输入的名字找到信息对应下标,要用到函数:FindName。然后再用新输入的数据覆盖老的数据,达到修改的效果。

        

2.10 排序通讯录信息:void SortContact(Contact* pc);

        通过冒泡排序,根据信息中的人名进行排序:

void SortContact(Contact* pc)
{
	assert(pc);
	//冒泡排序,按名字升序
	int i = 0;
	int j = 0;
	PeoInfo C = { 0 };
	for (i = 0;i < pc->count;i++)
	{
		for (j = 1;j < pc->count - i;j++)
		{
			if (strcmp(pc->data[j - 1].name, pc->data[j].name) > 0)
			{
				C = pc->data[j - 1];
				pc->data[j - 1] = pc->data[j];
				pc->data[j] = C;
			}
		}
	}
	printf("排序成功。");
}

        排序完成,输出:排序成功。

3 、效果展示

C语言:通讯录-LMLPHPC语言:通讯录-LMLPHPC语言:通讯录-LMLPHPC语言:通讯录-LMLPHPC语言:通讯录-LMLPHP

总结

        通过这样一个项目,可以很好地加强我们对结构体的运用,以及对C语言的掌握。希望这样的讲解对你有所帮助。

03-10 14:46