在实现这个通讯录之前先了解一下函数
1.sscanf
函数原型 int sscanf(const char *buffer, const char *format, [ argument ] … );
作用:从一个字符串中读进与指定格式相符的数据。
说明:sscanf()会将参数str的字符串根据参数format字符串来转换并格式化数据。转换后的结果存于对应的参数内。成功则返回参数数目,失败则返回0。sscanf与scanf类似,都是用于输入的,只是scanf以键盘(stdin)为输入源,sscanf以固定字符串为输入源。sscanf是一个很好用的函数,利用它可以从字符串中取出整数、浮点数和字符串等等。它的使用方法简单,特别对于整数和浮点数来说。
用法:<1>取指定长度的字符串。
char buf[512] = ;
sscanf("123456 ", "%s", buf);
printf("%s\n", buf);
输出结果为:123456
<2>取指定长度的字符串。如在下例中,取最大长度为4字节的字符串。
sscanf("123456 ", "%4s", buf);
printf("%s\n", buf);
结果为:1234
<3>取到指定字符为止的字符串。如在下例中,取遇到空格为止字符串。
sscanf("123456 abcdedf", "%[^ ]", buf);
printf("%s\n", buf);
结果为:123456
<4>取仅包含指定字符集的字符串。如在下例中,取仅包含1到9和小写字母的字符串。
sscanf("123456abcdedfBCDEF", "%[1-9a-z]", buf);
printf("%s\n", buf);
结果为:123456abcdedf
<5>取到指定字符集为止的字符串。如在下例中,取遇到大写字母为止的字符串。
sscanf("123456abcdedfBCDEF", "%[^A-Z]", buf);
printf("%s\n", buf);
结果为:123456abcdedf
<6>给定一个字符串iios/12DDWDFF@122,获取 / 和 @ 之间的字符串,先将 “iios/”过滤掉,再将非’@’的一串内容送到buf中
sscanf("iios/12DDWDFF@122", "%*[^/]/%[^@]", buf);
printf("%s\n", buf);
结果为:12DDWDFF
<7>给定一个字符串”hello, world”,仅保留”world”。(注意:“,”之后有一空格)
sscanf(“hello, world”, "%*s%s", buf);
printf("%s\n", buf);
结果为:world
2.sprintf
函数原型 int sprintf( char *buffer, const char *format, [ argument] … )
//把整数123 打印成一个字符串保存在s 中。
char s[10] = {0};
sprintf(s, "%d", 123); //产生"123"
- buffer:char型指针,指向将要写入的字符串的缓冲区。
- format:格式化字符串。
[argument]…:可选参数,可以是任何类型的数据。
返回写入buffer 的字符数,出错则返回-1. 如果 buffer 或 format 是空指针,且不出错而继续,函数将返回-1,并且 errno 会被设置为 EINVAL。
- sprintf 返回以format为格式argument为内容组成的结果被写入buffer 的字节数,结束字符‘\0’不计入内。即,如果“Hello”被写入空间足够大的buffer后,函数sprintf 返回5。同时buffer的内容将被改变。
3.fread
fread是一个函数,它从文件流中读数据,最多读取count个项,每个项size个字节,如果调用成功返回实际读取到的项个数(小于或等于count),如果不成功或读到文件末尾返回 0.
size_t fread ( void *buffer, size_t size, size_t count, FILE *stream) ;
- buffer
用于接收数据的内存地址
- size
要读的每个数据项的字节数,单位是字节 - count
要读count个数据项,每个数据项size个字节. - stream
输入流 - 返回真实读取的项数,若大于count则意味着产生了错误。另外,产生错误后,文件位置指示器是无法确定的。若其他stream或buffer为空指针,或在unicode模式中写入的字节数为奇数,此函数设置errno为EINVAL以及返回0.
- size
4.fwrite
fwrite() 是 C 语言标准库中的一个文件处理函数,功能是向指定的文件中写入若干数据块,如成功执行则返回实际写入的数据块数目。该函数以二进制形式对文件进行操作,不局限于文本文件。
size_t fwrite(const void* buffer, size_t size, size_t count, FILE* stream);
- 返回值:返回实际写入的数据块数目
- (1)buffer:是一个指针,对fwrite来说,是要获取数据的地址;
- (2)size:要写入内容的单字节数;
- (3)count:要进行写入size字节的数据项的个数;
- (4)stream:目标文件指针;
- (5)返回实际写入的数据项个数count。
contact.h
#ifndef CONTACT_H
#define CONTACT_H
enum Option
{
EXIT,
ADD,
DEL,
SEARCH,
MODIFY,
SHOW,
SORT,
EMPTY
};
#include<stdio.h>
#include<windows.h>
#include<stdlib.h>
#include<errno.h>
#define NAME_MAX 20
#define SEX_MAX 5
#define AGE_MAX 3
#define TEL_MAX 20
#define ADDR_MAX 30
#define MAX 1000
#define DEFAULT_SZ 3
#define FILENAME "Contact.dat"
typedef struct PeoInfo
{
char name[NAME_MAX];
char sex[SEX_MAX];
char age[AGE_MAX];
char tel[TEL_MAX];
char addr[ADDR_MAX];
}PeoInfo, *pPeoInfo;
typedef struct Contact
{
pPeoInfo data;//数据存放的位置
int sz;//有效元素的个数
int capacity;//容量
}Contact, *pContact;
void InitContact(pContact pc);//初始化通讯录
void DestroyContact(pContact pc);
void AddContact(pContact pc);//增添函数
void ShowContact(pContact pc); //展示函数
void DelContact(pContact pc);//删除函数
int SearchContact(pContact pc);//查找函数
int ModifyContact(pContact pc);//修改函数
void SortContact(pContact pc);//排序函数
int EmptyContact(pContact pc);//清空函数
#endif
contact.c
#define _CRT_SECURE_NO_WARNINGS 1
#include"contact.h"
void CheckCapacity(pContact pc)
{
if (pc->capacity == pc->sz)
{
pPeoInfo ptr = realloc(pc->data, (pc->capacity += 2)*sizeof(PeoInfo));
if (ptr != NULL)
{
pc->data = ptr;
}
printf("增容成功\n");
}
}
void LoadContact(pContact pc)
{
FILE *pfRead = fopen(FILENAME, "w");
PeoInfo tmp = { 0 };
if (pfRead == NULL)
{
perror("open file for read\n");
exit(EXIT_FAILURE);
}
//加载
while (fread(&tmp, sizeof(PeoInfo), 1, pfRead))
{
CheckCapacity(pc);
pc->data[pc->sz] = tmp;
pc->sz++;
}
free(pfRead);
pfRead = NULL;
}
void InitContact(pContact pc)
{
pc->sz = 0;
pc->capacity = DEFAULT_SZ;
pc->data = (pPeoInfo)calloc(pc->capacity, sizeof(PeoInfo));
if (pc->data == NULL)
{
printf("%s\n", strerror(errno));
exit(EXIT_FAILURE);//失败退出程序
//加载已有文件信息
LoadContact(pc);
}
}
void SaveContact(pContact pc)
{
unsigned int i = 0;
FILE* pfWrite = fopen(FILENAME, "w");
if (pfWrite == NULL)
{
perror("open file for write\n");
exit(EXIT_FAILURE);
}
for (i = 0; i < pc->sz; i++)
{
fwrite(pc->data + i, sizeof(PeoInfo), 1, pfWrite);
}
free(pfWrite);
pfWrite = NULL;
}
void DestroyContact(pContact pc)
{
SaveContact(pc);
free(pc->data);
pc->data = NULL;
pc->capacity = 0;
pc->sz = 0;
}
void AddContact(pContact pc)
{
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].tel);
printf("请输入地址:");
scanf("%s", pc->data[pc->sz].addr);
pc->sz++;
printf("添加成功\n");
}
void ShowContact(pContact pc)
{
unsigned int i = 0;
printf("%-10s\t %-5s\t %-5s %-10s %-10s\n", "name", "sex", "age", "tel", "addr");
for (i = 0; i < pc->sz; i++)
{
printf("%-10s\t %-5s\t %-5s %-10s %-10s\n", pc->data[i].name, pc->data[i].sex, pc->data[i].age, pc->data[i].tel, pc->data[i].addr);
}
}
static int FindEntry(pContact pc, char name[])
{
unsigned int i = 0;
for (i = 0; i < pc->sz; i++)
{
if (strcmp(name, pc->data[i].name) == 0)
{
return i;
};
return -1;
}
}
void DelContact(pContact pc)
{
char name[NAME_MAX] = { 0 };
int pos = 0;
if (pc->sz == 0)
{
printf("没有信息可以删除\n");
return;
}
printf("请输入要删除人的名字:>");
scanf("%s", name);
FindEntry(pc, name);
if (pos == -1)
{
printf("删除的人不存在\n");
return;
}
else
{
for (int i = 0; i < pc->sz - 1; i++)
{
pc->data[i] = pc->data[i + 1];
}
pc->sz--;
printf("删除成功\n");
}
}
int SearchContact(pContact pc)
{
int ret = FindEntry(pc, pc->data);
if (ret != -1)
{
printf("该联系人的信息为:\n");
printf("姓名:%-5s\n", pc->data[ret].name, NAME_MAX);
printf("性别:%-5s\n", pc->data[ret].sex, SEX_MAX);
printf("年龄:%-5s\n", pc->data[ret].sex, SEX_MAX);
printf("电话:%-5s\n", pc->data[ret].tel, TEL_MAX);
printf("地址:%-5s\n", pc->data[ret].addr, ADDR_MAX);
return 1;
}
else
{
printf("查不到该联系人\n");
return -1;
}
}
int ModifyContact(pContact pc)
{
int ret = FindEntry(pc, pc->data);//定义ret 接收find_entry的返回位置
if (ret != -1)
{
printf("请重新编辑该联系人信息:\n");
printf("请输入姓名:");
scanf_s("%s", pc->data[ret].name, NAME_MAX);
printf("请输入性别:");
scanf_s("%s", pc->data[ret].sex, SEX_MAX);
printf("请输入年龄:");
scanf_s("%s", pc->data[ret].sex, SEX_MAX);
printf("请输入电话:");
scanf_s("%s", pc->data[ret].tel, TEL_MAX);
printf("请输入地址:");
scanf_s("%s", pc->data[ret].addr, ADDR_MAX);
printf("修改成功!");
return 1;
}
else
{
printf("找不到该联系人\n");
return -1;
}
}
void SortContact(pContact pc)
{
int i = 0;
int j = 0;
for (i = 0; i < pc->sz - 1; i++)
{
for (j = 0; j < pc->sz - 1 - i; j++)
{
if (strcmp(pc->data[j].name, pc->data[j + 1].name)>0)
{
struct PeoInfo tmp = pc->data[j];
pc->data[j] = pc->data[j + 1];
pc->data[j + 1] = tmp;
}
}
}
printf("排序完成\n");
}
int EmptyContact(pContact pc)
{
pc->sz = 0;//count 赋值0 进行清零
printf("清空成功!");
return 1;
}
test.c
#define _CRT_SECURE_NO_WARNINGS 1
#include"contact.h"
void menu()
{
printf("------------ 服务菜单 ---------------\n");
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");
printf("-------------------------------------\n");
}
int main()
{
int input = 0;
struct Contact my_con;
InitContact(&my_con);
do
{
menu();
printf("请选择:>");
scanf_s("%d", &input);
switch (input)
{
case ADD:
AddContact(&my_con);
break;
case DEL:
DelContact(&my_con);
break;
case SEARCH:
SearchContact(&my_con);
break;
case MODIFY:
ModifyContact(&my_con);
break;
case SHOW:
ShowContact(&my_con);
break;
case SORT:
SortContact(&my_con);
break;
case EMPTY:
EmptyContact(&my_con);
break;
case EXIT:
DestroyContact(&my_con);
break;
default:
printf("选择错误\n");
break;
}
} while (input);
return 0;
}