コンテンツ
静的バージョンはアドレス帳を実装します:(配列のサイズは変更できません)
静的に開発された名簿リストは1000に固定されており、無駄なスペースが少なく、より多くの再変更が必要であることがわかりました。動的メモリ開発を使用して名簿を初期化できますか?
プログラムを終了すると、アドレス帳が破棄されます。入力したリストをファイルに保存して、簡単に追加して表示できますか?
名簿がデータ構造に移行しているのはなぜですか?2つの理由があります:
1.このブログを書いた後、データ構造学習に入ります
2.名簿は、C言語の学習の概要です。データ構造を使用してC言語を実装するには、しっかりとした基盤が必要なため、自分で名簿を作成することをお勧めします。名簿を作成することは、知識を十分に理解していることを意味します。
コードを標準化し、モジュール式にコードを記述するために、複数ファイルのコンパイルを使用します
主な機能の実装 | test.c |
関数宣言 | Contact.h |
関数の実装 | Contact.c |
静的バージョンはアドレス帳を実装します:(配列のサイズは変更できません)
1.名前、性別、電話番号、年齢、および名簿に保存された情報数の記録を含む名簿構造を作成します(追加、削除時に名簿リストを簡単に変更できるようにするため) 、チェック、および変更)
2.ヘッダーファイルが複数回インクルードされないように、ヘッダーファイルをヘッダーファイルにインクルードする必要があります
3. main関数を実装するときは、main関数本体を作成する必要があります。コードを標準化するために、列挙型を作成し、列挙型を使用してコードを記述できます。
test.cボディ構造
enum contact
{
EXIT,//0
ADD,
DEL,
SEARCH,
MODIFY,
SORT,
PRINT,
};
void menu()
{
printf("*****************************\n");
printf("**** 1.ADD 2.DEL ********\n");
printf("**** 3.SEARCH 4.MODIFY ******\n");
printf("**** 5.SORT 6.PRINT ********\n");
printf("*****0.EXIT ********\n");
printf("*****************************\n");
}
void test()
{
int input = 0;
do
{
menu();
printf("请选择->");
scanf("%d", &input);
switch (input)
{
case EXIT:
break;
case ADD:
break;
case DEL:
case SEARCH:
break;
case MODIFY:
break;
case SORT:
break;
case PRINT:
break;
}
} while (input);
}
int main()
{
test();
return 0;
}
4.ヘッダーファイルの実装では、列挙をヘッダーファイルに入れることができます。ヘッダーファイルでは、変更を容易にするために#defineが使用されます。サイズなどのアドレスブックリストを拡大および縮小する場合は、名前配列、それは非常に便利です、定義を変更するだけです
Contact.hのボディ構造
//通讯录实现 接口申明
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>
#define MAX 1000
#define NAME 20
#define SEX 5
#define PHONE 12
#define ADDR 30
#define DEFAULF_SZ 3
typedef struct PeoInfo
{
char name[NAME];//姓名
char sex[SEX];//性别
int age;//年龄
char phone[PHONE];//电话
char addr[ADDR];//地址
}PeoInfo;
enum contact
{
EXIT,
ADD,
DEL,
SEARCH,
MODIFY,
SORT,
PRINT,
};
静态版本
typedef struct Contact
{
PeoInfo data[MAX];
int sz;
}Contact;
必要な機能を宣言する
// 初始化通讯录
void InitContact(Contact* pc);
//销毁通讯录
void DestoryContact(Contact* pc);
//添加通讯录名单
void Addcontact(Contact * pc);
//打印通讯录名单
void PrintContact(const Contact* pc);
//删除通讯录名单
void DelContact(Contact* pc);
//查找通讯录名单
void SearchContact(const Contact* pc);
//改通讯录名单
void ModifyContact(Contact* pc);
//保存通讯录到文件
void SaveContact(const Contact* pc);
5.アドレス帳に必要な機能の実装方法は、アドレス帳を作成した後、まずアドレス帳を初期化する必要があります(メイン関数が呼び出されたときに作成された後に初期化されます)
Contact.c関数は、アドレス帳の初期化を実装し、最初に配列の内容を0に初期化します。
静态版本
void InitContact(Contact* pc)//初始化通讯录
{
assert(pc);
memset(pc->data,0,sizeof(pc->data));
pc->sz = 0;
}
Contact.c関数は、アドレス帳リストの追加を実装します。配列szの添え字がfull = 1000の場合は、入力を停止します。
void Addcontact(Contact* pc)
{
assert(pc);
if (pc->sz == MAX)
{
printf("通讯录已满,无法添加\n");
return;
}
printf("请输入姓名");
scanf("%s", pc->data[pc->sz].name);
printf("请输入性别");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入年龄");
scanf("%d", &(pc->data[pc->sz].age));
printf("请输入手机号");
scanf("%s", pc->data[pc->sz].phone);
printf("请输入地址");
scanf("%s", pc->data[pc->sz].addr);
pc->sz++;
printf("添加成功!\n");
}
Contact.c関数は、アドレスブックリストの検索、新しい配列への名前の入力、検索のトラバース、および返された配列のsz添え字の検索を実現します。
int FindContact(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;
}
Contact.c関数は、アドレスブックリストの削除、新しい配列への名前の入力、および検索のトラバースを実装します。最後の要素はカバーされていませんが、sz--の後には見つかりません。
void DelContact(Contact* pc)
{
assert(pc);
if (pc->sz == 0)
{
printf("通讯录已空,无法删除\n");
return;
}
//找到要删除的人
printf("请输入要删除人的信息\n");
char name[NAME] = {0};
scanf("%s", name);
int pos=FindContact(pc,name);
if (pos == -1)
{
printf("找不到\n");
return;
}
int j = 0;
for (j = pos; j < pc->sz-1; j++)
{
pc->data[j] = pc->data[j + 1];
}
pc->sz--;
printf("删除成功\n");
}
名簿リストを印刷するContact.c関数
void SearchContact(const Contact* pc)
{
char name[NAME] = { 0 };
printf("请输入要查找人的名字");
scanf("%s", name);
int pos = FindContact(pc, name);
if (pos == -1)
{
printf("找不到\n");
return;
}
printf("%-20s %-5s %-3s %-12s %-30s\n", "姓名", "性别", "年龄", "手机号", "地址");
printf("%-20s %-5s %-3d %-12s %-30s\n", pc->data[pos].name, pc->data[pos].sex, pc->data[pos].age, pc->data[pos].phone, pc->data[pos].addr);
}
Contact.c関数は、アドレス帳リストの変更を実装します
void ModifyContact(Contact* pc)
{
char name[NAME] = { 0 };
printf("请输入要查找人的名字");
scanf("%s", name);
int pos = FindContact(pc, name);
if (pos == -1)
{
printf("找不到\n");
return;
}
else
{
printf("请输入姓名");
scanf("%s", pc->data[pc->sz].name);
printf("请输入性别");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入年龄");
scanf("%d", &(pc->data[pc->sz].age));
printf("请输入手机号");
scanf("%s", pc->data[pc->sz].phone);
printf("请输入地址");
scanf("%s", pc->data[pc->sz].addr);
printf("修改成功!");
}
}
動的バージョンはアドレス帳を実装します
静的に開発された名簿リストは1000に固定されており、無駄なスペースが少なく、より多くの再変更が必要であることがわかりました。動的メモリ開発を使用して名簿を初期化できますか?
//动态版本
void InitContact(Contact* pc)
{
assert(pc);
//memset(pc->data,0,sizeof(pc->data));
pc->sz = 0;
pc->capacity = DEFAULF_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));
}
增容通讯录
void CheckCapacity(Contact* pc)
{
if (pc->capacity == pc->sz)
{
//增容
PeoInfo* tmp = (PeoInfo*)realloc(pc->data, (pc->capacity + 2) * sizeof(PeoInfo));//realloc调整后的总大小
if (tmp != NULL)
{
pc->data = tmp;
}
else
{
perror("CheckCapacity::realloc");
return;
}
pc->capacity += 2;
printf("增容成功\n");
}
}
動的メモリ開発を使用してアドレス帳を初期化したため、アドレス帳リストを追加するときにアドレス帳がいっぱいかどうかも確認し、いっぱいになったときに容量を増やす必要があります。
ファイルバージョンはアドレスブックを実装します
プログラムを終了すると、アドレス帳が破棄されます。入力したリストをファイルに保存して、簡単に追加して表示できますか?
ファイルを開いた後、アドレス帳を破棄する必要があり、プログラムで書き込まれたアドレス帳がファイルに保存されます
void DestoryContact(Contact* pc)
{
free(pc->data);
pc->data = NULL;
pc->capacity = 0;
pc->sz = 0;
printf("销毁成功\n");
}
void SaveContact(const Contact* pc)
{
FILE* pf =fopen("Contact.txt", "wb");//二进制方式写进去
if (pf == NULL)
{
perror("SaveContact::fopen");
return ;
}
//写文件
int i = 0;
for (i = 0; i < pc->sz; i++)
{
fwrite(pc->data + i, sizeof(PeoInfo), 1, pf);
}
printf("保存成功!\n");
//关闭文件
fclose(pf);
pf = NULL;
}
名簿の初期化-ファイルバージョン
最後に保存したアドレス帳リストをプログラムにインポートしてロードします(初期化時)
//初始化通讯录 --文件版本
void LoadContact(Contact* pc)
{
//打开文件
FILE*pf =fopen("Contact.txt", "rb");
if (pf == NULL)
{
perror("LoadContact::fopen");
return;
}
//读文件
PeoInfo tmp = { 0 };
while (fread(&tmp, sizeof(PeoInfo), 1, pf))//返回的数值比要求读写的count数据小,代表读写完成,如果返回0代表没读取到
{
//通讯录初始化也要扩容,防止数据大于初始值
CheckCapacity(pc);
pc->data[pc -> sz] = tmp;//读取一个信息放到sz中,因为默认sz=0
pc->sz++;
}
//关闭文件
fclose(pf);
pf = NULL;
}
void InitContact(Contact* pc)
{
assert(pc);
pc->sz = 0;
pc->capacity = DEFAULF_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));
//加载文件信息到通讯录中
LoadContact(pc);
}
完全なContact.hヘッダーファイルの実装宣言
//通讯录实现 接口申明
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <assert.h>
#include <stdlib.h>
#define MAX 10
#define NAME 20
#define SEX 5
#define PHONE 12
#define ADDR 30
#define DEFAULF_SZ 3
typedef struct PeoInfo
{
char name[NAME];
char sex[SEX];
int age;
char phone[PHONE];
char addr[ADDR];
}PeoInfo;
enum contact
{
EXIT,
ADD,
DEL,
SEARCH,
MODIFY,
SORT,
PRINT,
};
//静态版本
//typedef struct Contact
//{
// PeoInfo data[MAX];
// int sz;
//}Contact;
//动态版本
typedef struct Contact
{
PeoInfo *data;//可以存放动态人数
int sz;//记录通讯录中已经保存的信息个数
int capacity;//记录通讯录当前的最大容量
}Contact;
// 初始化通讯录
void InitContact(Contact* pc);
//销毁通讯录
void DestoryContact(Contact* pc);
//添加通讯录名单
void Addcontact(Contact * pc);
//打印通讯录名单
void PrintContact(const Contact* pc);
//删除通讯录名单
void DelContact(Contact* pc);
//查找通讯录名单
void SearchContact(const Contact* pc);
//改通讯录名单
void ModifyContact(Contact* pc);
//保存通讯录到文件
void SaveContact(const Contact* pc);
完全なtest.cソースファイル実装フレームワーク
//专门测试通讯录功能
#include "contact.h"
void menu()
{
printf("*****************************\n");
printf("**** 1.ADD 2.DEL ********\n");
printf("**** 3.SEARCH 4.MODIFY ******\n");
printf("**** 5.SORT 6.PRINT ********\n");
printf("*****0.EXIT ********\n");
printf("*****************************\n");
}
void test()
{
Contact con;//创建通讯录
InitContact(&con);//初始化通讯录
int input = 0;
do
{
menu();
printf("请选择->");
scanf("%d", &input);
switch (input)
{
case EXIT:
SaveContact(&con);
DestoryContact(&con);
break;
case ADD:
Addcontact(&con);
break;
case DEL:
DelContact(&con);
case SEARCH:
SearchContact(&con);
break;
case MODIFY:
ModifyContact(&con);
break;
case SORT:
break;
case PRINT:
PrintContact(&con);
break;
}
} while (input);
}
int main()
{
test();
return 0;
}
完全なContact.cソースファイル実装機能
//通讯录实现 接口(函数)实现
#include "contact.h"
//静态版本
//void InitContact(Contact* pc)
//{
// assert(pc);
// memset(pc->data,0,sizeof(pc->data));
// pc->sz = 0;
//}
//动态版本
//void InitContact(Contact* pc)
//{
// assert(pc);
// //memset(pc->data,0,sizeof(pc->data));
// pc->sz = 0;
// pc->capacity = DEFAULF_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));
//}
//
//增容通讯录
void CheckCapacity(Contact* pc)
{
if (pc->capacity == pc->sz)
{
//增容
PeoInfo* tmp = (PeoInfo*)realloc(pc->data, (pc->capacity + 2) * sizeof(PeoInfo));//realloc调整后的总大小
if (tmp != NULL)
{
pc->data = tmp;
}
else
{
perror("CheckCapacity::realloc");
return;
}
pc->capacity += 2;
printf("增容成功\n");
}
}
//初始化通讯录 --文件版本
void LoadContact(Contact* pc)
{
//打开文件
FILE*pf =fopen("Contact.txt", "rb");
if (pf == NULL)
{
perror("LoadContact::fopen");
return;
}
//读文件
PeoInfo tmp = { 0 };
while (fread(&tmp, sizeof(PeoInfo), 1, pf))//返回的数值比要求读写的count数据小,代表读写完成,如果返回0代表没读取到
{
//通讯录初始化也要扩容,防止数据大于初始值
CheckCapacity(pc);
pc->data[pc -> sz] = tmp;//读取一个信息放到sz中,因为默认sz=0
pc->sz++;
}
//关闭文件
fclose(pf);
pf = NULL;
}
void InitContact(Contact* pc)
{
assert(pc);
pc->sz = 0;
pc->capacity = DEFAULF_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));
//加载文件信息到通讯录中
LoadContact(pc);
}
void Addcontact(Contact* pc)
{
assert(pc);
//静态版本
/*if (pc->sz == MAX)
{
printf("通讯录已满,无法添加\n");
return;
}*/
//动态版本
CheckCapacity(pc);
//录入信息
printf("请输入姓名");
scanf("%s", pc->data[pc->sz].name);
printf("请输入性别");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入年龄");
scanf("%d", &(pc->data[pc->sz].age));
printf("请输入手机号");
scanf("%s", pc->data[pc->sz].phone);
printf("请输入地址");
scanf("%s", pc->data[pc->sz].addr);
pc->sz++;
printf("添加成功!\n");
}
void PrintContact(const Contact* pc)
{
assert(pc);
int i = 0;
printf("%-20s %-5s %-3s %-12s %-30s\n", "姓名", "性别", "年龄", "手机号", "地址");
for (i = 0; i < pc->sz; i++)
{
printf("%-20s %-5s %-3d %-12s %-30s\n", pc->data[i].name, pc->data[i].sex, pc->data[i].age, pc->data[i].phone, pc->data[i].addr);
}
}
int FindContact(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;
}
//找到要删除的人
printf("请输入要删除人的信息\n");
char name[NAME] = {0};
scanf("%s", name);
int pos=FindContact(pc,name);
if (pos == -1)
{
printf("找不到\n");
return;
}
int j = 0;
for (j = pos; j < pc->sz-1; j++)
{
pc->data[j] = pc->data[j + 1];
}
pc->sz--;
printf("删除成功\n");
}
void SearchContact(const Contact* pc)
{
char name[NAME] = { 0 };
printf("请输入要查找人的名字");
scanf("%s", name);
int pos = FindContact(pc, name);
if (pos == -1)
{
printf("找不到\n");
return;
}
printf("%-20s %-5s %-3s %-12s %-30s\n", "姓名", "性别", "年龄", "手机号", "地址");
printf("%-20s %-5s %-3d %-12s %-30s\n", pc->data[pos].name, pc->data[pos].sex, pc->data[pos].age, pc->data[pos].phone, pc->data[pos].addr);
}
void ModifyContact(Contact* pc)
{
char name[NAME] = { 0 };
printf("请输入要查找人的名字");
scanf("%s", name);
int pos = FindContact(pc, name);
if (pos == -1)
{
printf("找不到\n");
return;
}
else
{
printf("请输入姓名");
scanf("%s", pc->data[pc->sz].name);
printf("请输入性别");
scanf("%s", pc->data[pc->sz].sex);
printf("请输入年龄");
scanf("%d", &(pc->data[pc->sz].age));
printf("请输入手机号");
scanf("%s", pc->data[pc->sz].phone);
printf("请输入地址");
scanf("%s", pc->data[pc->sz].addr);
printf("修改成功!");
}
}
void DestoryContact(Contact* pc)
{
free(pc->data);
pc->data = NULL;
pc->capacity = 0;
pc->sz = 0;
printf("销毁成功\n");
}
void SaveContact(const Contact* pc)
{
FILE* pf =fopen("Contact.txt", "wb");//二进制方式写进去
if (pf == NULL)
{
perror("SaveContact::fopen");
return ;
}
//写文件
int i = 0;
for (i = 0; i < pc->sz; i++)
{
fwrite(pc->data + i, sizeof(PeoInfo), 1, pf);
}
printf("保存成功!\n");
//关闭文件
fclose(pf);
pf = NULL;
}