通讯录二(动态开辟)

contact.h

#pragma once
#include<stdio.h>
#include<string.h>
#include<assert.h>
#include<stdlib.h>
//类型的声明

#define Max 1000
#define Name_Max 20
#define Sex_Max 5
#define Tele_Max 12
#define add_Max 30
#define DEFAULT_SZ 3

//这样使下面的选择代码更有可读性
enum Option {
	EXIT,
	ADD,
	DEL,
	SEARCH,
	MODIFY,
	SORT,
	PRINT,
	CLEAR
};

typedef struct PeoInfo {
	char name[Name_Max];
	char sex[Sex_Max];
	int age;
	char tele[Tele_Max];
	char add[add_Max];
}PeoInfo;

//通讯录结构体,通过封装可以更好的传参,完成增删改查
//typedef struct Contact {
//	//创建10000人信息
//	PeoInfo data[Max];
//	//个数
//	int sz;
//}Contact;

//建立动态版本
typedef struct Contact {
	//创建10000人信息
	PeoInfo* data;
	int sz;
	int capacity;//记录通讯录当前的最大容量
}Contact;

//函数声明
//增加
void AddContact(Contact* pc);
//初始化
void InitContact(Contact* pc);
//打印
void PrintContact(const Contact* pc);
//删除指定联系人
void DelContact(Contact* pc);
//查找联系人
void SearchContact(const Contact* pc);
//修改指定联系人
void ModifyContact(Contact* pc);
//按名字进行排序
void SortContact(Contact* pc);
//清空通讯表
void ClearContact(Contact* pc);
//销毁通讯录
void DestroyContact(Contact* pc);

contact.c

#define _CRT_SECURE_NO_WARNINGS
#include "contact.h"

checkCapacity(Contact* pc) {
	if (pc->sz == pc->capacity) {
		PeoInfo* tmp = realloc(pc->data, (pc->capacity + 2) * sizeof(PeoInfo));
		if (tmp != NULL) {
			pc->data = tmp;
		}
		else {
			perror("InitContact::malloc");
			return;
		}
		pc->capacity += 2;
		printf("增容成功!\n");
	}
}

//增加信息
void AddContact(Contact* pc) {
	assert(pc);
	//录入信息
	/*if (pc -> sz == Max) {
		printf("通讯录已满,无法添加\n");
		return;
	}*/
	//动态版本
	checkCapacity(pc);

	printf("请输入名字:》\n");
	scanf("%s", pc->data[pc->sz].name);
	printf("请输入年龄:》\n");
	scanf("%d", &(pc->data[pc->sz].age));//记得取地址
	printf("请输入性别:》\n");
	scanf("%s", pc->data[pc->sz].sex);
	printf("请输入电话:》\n");
	scanf("%s", pc->data[pc->sz].tele);
	printf("请输入地址:》\n");
	scanf("%s", pc->data[pc->sz].add);

	pc->sz++;
	printf("添加成功!\n");

}
//初始化
//void InitContact(Contact* pc) {//通过指针找到地址,再找到里面的成员进行计算大小而不是计算地址大小
//	assert(pc);
//	pc->sz = 0;
//	memset(pc->data,0,sizeof(pc->data));//实在是妙!没想到用memset
//}

//动态的
void InitContact(Contact* pc) {//通过指针找到地址,再找到里面的成员进行计算大小而不是计算地址大小
	assert(pc);
	pc->sz = 0;
	pc->capacity = DEFAULT_SZ;
	pc->data = (PeoInfo*)malloc(pc->capacity*sizeof(PeoInfo));
	if (pc->data == NULL) {
		perror("InitContact::malloc");
		return;
	}
	memset(pc->data, 0, pc->capacity * sizeof(PeoInfo));//实在是妙!没想到用memset
}

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

//找到返回下标,否则-1
int FindByName(const Contact* pc, char name[]) {
	assert(pc);
	int i = 0;
	for (i = 0; i < pc->sz; i++) {
		if (strcmp(pc->data[i].name, name) == 0) {
			return i;
		}
	}
	return -1;
}

void DelContact(Contact* pc) {
	assert(pc);
	if (pc->sz == 0) {
		printf("通讯录已经空了!\n");
		return;
	}
	//找人
	char name[Name_Max] = { 0 };
	printf("请输入要删除的人:》\n");
	scanf("%s", name);
	int pos=FindByName(pc, name);
	if (pos == -1) {
		printf("不存在!\n");
	}
	
	int j = 0;
	if (pos != pc->sz - 1) {
		for (j = pos; j < pc->sz - 1; j++) {
			pc->data[j] = pc->data[j + 1];
		}
	}
	pc->sz--;
	printf("删除成功!\n");

}

void SearchContact(const Contact* pc) {
	assert(pc);
	//找人
	char name[Name_Max] = { 0 };
	printf("请输入要查找的人:》\n");
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1) {
		printf("不存在!\n");
	}
	printf(" % -10s % -5s % -5s % -12s % -10s\n", "姓名", "年龄", "性别", "电话", "地址");
	printf("%-10s  %-5d %-5s %-12s %-10s\n", pc->data[pos].name, pc->data[pos].age, pc->data[pos].sex, pc->data[pos].tele, pc->data[pos].add);

}

void ModifyContact(Contact* pc) {
	assert(pc);
	printf("输入要修改的人:>");
	char name[Name_Max] = { 0 };
	scanf("%s", name);
	int pos = FindByName(pc, name);
	if (pos == -1) {
		printf("不存在!\n");
	}
	printf("请输入年龄:》\n");
	scanf("%d", &(pc->data[pos].age));//记得取地址
	printf("请输入性别:》\n");
	scanf("%s", pc->data[pos].sex);
	printf("请输入电话:》\n");
	scanf("%s", pc->data[pos].tele);
	printf("请输入地址:》\n");
	scanf("%s", pc->data[pos].add);
	printf("修改成功!\n");

}

static int cmp(const void* p1, const void* p2) {
	return strcmp(((PeoInfo*)p1)->name, ((PeoInfo*)p2)->name);
}

void SortContact(Contact* pc) {
	assert(pc);
	if (pc->sz == 0) {
		printf("通讯表为空!\n");
	}
	qsort(pc->data,pc->sz,sizeof(pc->data[0]),cmp);
	printf("排序成功!\n");
}

void ClearContact(Contact* pc) {
	assert(pc);
	if (pc->sz == 0) {
		printf("通讯表为空,无需清空!\n");
		return;
	}
	//初始化通讯录
	InitContact(pc);
	printf("清空通讯录!\n");
}

void DestroyContact(Contact* pc) {
	free(pc->data);
	pc->data = NULL;
	pc->capacity = 0;
	pc->sz = 0;
	printf("销毁成功!\n");
}

test.c

#define _CRT_SECURE_NO_WARNINGS
#include "contact.h"


void menu() {
	printf("---------------------\n");
	printf("---1.add---2.del-----\n");
	printf("---3.serch 4.modify--\n");
	printf("---5.sort- 6.printf --\n");
	printf("---7.clear 0.exit----\n");
	printf("---------------------\n");
}

void test() {
	int input = 0;
	//创建通讯录
	Contact con;

	//初始化
	InitContact(&con);

	do {
		menu();
		printf("请选择:》\n");
		scanf("%d", &input);
		switch (input) {
		case ADD:
			AddContact(&con);
			break;
		case DEL:
			DelContact(&con);
			break;
		case SEARCH:
			SearchContact(&con);
			break;
		case MODIFY:
			ModifyContact(&con);
			break;
		case SORT:
			SortContact(&con);
			break;
		case PRINT:
			PrintContact(&con);
			break;
		case CLEAR:
			ClearContact(&con);
			break;
		case EXIT:
			DestroyContact(&con);
			printf("退出通讯录\n");
			break;
		default:
			printf("选择错误\n");
			break;
		}
	} while(input);
}

int main() {
	test();
	return 0;
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值