C語(yǔ)言通過(guò)三種方法實(shí)現(xiàn)屬于你的通訊錄
一、基礎(chǔ)版本
前提準(zhǔn)備:
1、通訊錄里面的人的個(gè)人信息(姓名、性別、年齡、手機(jī)號(hào)、地址)。
2、通訊錄名單(來(lái)實(shí)現(xiàn)人員的增刪插改)。
3、人員的初始化。
4、菜單。
1.1 通訊錄的個(gè)人信息(結(jié)構(gòu)體來(lái)實(shí)現(xiàn))
typedef struct peoInfo //重定義
{
char name[20]; //姓名
int age ; //年齡
char sex[10]; //性別
char tele[12]; //手機(jī)號(hào)
char addr[30]; //地址
}peoInfo;
1.2通訊錄名單
typedef struct Contact
{
peoInfo data[100]; //人員名單 結(jié)構(gòu)體數(shù)組
int count; //人員數(shù)量
}Contact;
1.3人員初始化
void InitContact(Contact* con)
{
assert(con);
con->count=0; //初始人數(shù)為0
memset(con->data,0,sizeof(con->data)); //這是一個(gè)c語(yǔ)言的庫(kù)函數(shù)把數(shù)組里面的數(shù)全部初始化為0
}
1.4菜單
void menu(void)
{
printf("===============================\n");
printf("====1.add 2. del 3.serch=====\n");
printf("=====4.modify 5.show======\n");
printf("=====6.sort 0.exit======\n");
printf("===============================\n");
printf("===============================\n");
}
1.5主函數(shù)
enum a
{
ext, //這里用了簡(jiǎn)單的枚舉的方法 枚舉的初始值為0
add,
del,
serch,
modify,
show,
sort
};
int main()
{
int choice;
Contact con; //通訊錄
do
{
menu();
printf("請(qǐng)選擇功能\n");
scanf("%d",&choice);
switch(choice)
{
case add:
AddContact(&con); //實(shí)現(xiàn)增加功能的函數(shù)
break ;
case del:
DelContanct(&con); //實(shí)現(xiàn)刪除功能的函數(shù)
break ;
case serch:
SerContact(&con); //實(shí)現(xiàn)查找功能的函數(shù)
break ;
case modify:
ModContact(& con); //實(shí)現(xiàn)修改功能的函數(shù)
break ;
case show:
ShowContact(&con); //實(shí)現(xiàn)展示功能的函數(shù)
break ;
case sort:
SorContact(&con); //實(shí)現(xiàn)排序功能的函數(shù)
break ;
case ext :
printf("程序結(jié)束\n");
break;
default :
printf("輸入有誤請(qǐng)重新輸入\n");
}
}while(choice);
return 0;
}二、功能的實(shí)現(xiàn)
2.1、增加人數(shù)
void AddContact(Contact* con)
{
assert(con); //指針不能為空
if(con->count==Max)
{
printf("存放已滿(mǎn)\n");
return ;
}
printf("請(qǐng)輸入名字");
scanf("%s",con->data[con->count].name);
printf("請(qǐng)輸入年齡");
scanf("%d",&(con->data[con->count].age));
printf("請(qǐng)輸入性別");
scanf("%s",con->data[con->count].sex);
printf("請(qǐng)輸入電話");
scanf("%s",con->data[con->count].tele);
printf("請(qǐng)輸入地址");
scanf("%s",con->data[con->count].addr);
con->count++;
printf("增加成功\n");
}2.2、刪除人數(shù)
static int xiaomafind(Contact* con,char name[])
{
assert(con);
int i=0;
for(i=0;i<con->count;i++)
{
if(0==strcmp(con->data[i].name,name)) //通過(guò)名字來(lái)查找看是否存在 ,如果有返回下標(biāo)。
{
return i;
}
else
{
return -1;
}
}
void DelContanct(Contact* con)
{
char name[20]={0};
printf("請(qǐng)輸入你想刪除的名字\n");
assert(con);
int i;
scanf("%s",name);
int pos=xiaomafind(con,name);
if(pos==-1)
{
printf("沒(méi)有你要?jiǎng)h除的人\n");
}
else
{
for(i=pos;i<con->count;i++)
{
con->data[i]=con->data[i+1]; //通過(guò)數(shù)組后一個(gè)覆蓋前一個(gè)來(lái)完成刪除功能
}
con->count--; //刪除完成后人員減一
printf("刪除成功\n");
}
}2.3、查找
static int xiaomafind(Contact* con,char name[])
{
assert(con);
int i=0;
for(i=0;i<con->count;i++)
{
if(0==strcmp(con->data[i].name,name))
{
return i;
}
}
return -1;
}
void SerContact(Contact *con)
{
char name[20]={0};
printf("請(qǐng)輸入你想查找的名字\n");
assert(con);
scanf("%s",name);
int find =xiaomafind(con,name);
if(find==-1)
{
printf("沒(méi)有你要查找的用戶(hù)\n");
}
else{
printf("查找成功\n");
}
}2.4、展示
void ShowContact( const Contact* con)
{
assert(con);
int i;
for(i=0;i<con->count;i++)
{
printf("%2s\t%3d\t%5s\t%12s\t%30s\n",con->data[i].name,con->data[i].age,con->data[i].sex,con->data[i].tele,con->data[i].addr); //這里是為了打印好看
}
}
2.5、排序(這里我是通過(guò)名字)
^?_?^用qsort 函數(shù)發(fā)現(xiàn)很很容的都能進(jìn)行排序,我們只需要告訴他我們的排序方法是什么就可以很容易的進(jìn)行一下排序
int cmp_stu_by_name(const void*e1,const void*e2)
{
return strcmp(((peoInfo*)e1)->name,((peoInfo*)e2)->name);
} //如不不理解的看我指針講解中回調(diào)函數(shù)
void SorContact(Contact* con)
{
assert(con);
qsort(con->data, con->count, sizeof(peoInfo), cmp_stu_by_name); //這里運(yùn)用了一個(gè)c語(yǔ)言的庫(kù)函數(shù)qsort函數(shù),在指針進(jìn)階中我也講述了如何使用,如果不會(huì)的可以看一下指針的進(jìn)階。
}
三、通訊錄進(jìn)階(設(shè)置動(dòng)態(tài)存儲(chǔ))
動(dòng)態(tài)版本只需要改三個(gè)位置就行啦
3.1通訊錄從靜態(tài)改為動(dòng)態(tài)
//靜態(tài)版本
//typedef struct Contact
//{
// peoInfo data[100]; //人員名單 結(jié)構(gòu)體數(shù)組
// int count; //人員數(shù)量
//}Contact;
//動(dòng)態(tài)版本
???????typedef struct Contact
{
peoInfo *data; //人員名單 結(jié)構(gòu)體指針
int count; //人員數(shù)量
int capicity; //通訊錄容量
}Contact;
3.2通訊錄的初始化
//靜態(tài)版本
//void InitContact(Contact* con)
//{
// assert(con);
// con->count=0; //初始人數(shù)為0
// memset(con->data,0,sizeof(con->data)); //這是一個(gè)c語(yǔ)言的庫(kù)函數(shù)把數(shù)組里面的數(shù)全部初始化為0
//}
//動(dòng)態(tài)版本
void InitContact(Contact* con)
{
assert(con);
con->count=0;
con->data=(PeoInf0*)calloc(3,sizeof(peoInfo*));
if(con->data=NULL)
{
perrror("con->data"); //如果空指針打印錯(cuò)誤原因
}
con->capicity=3; //這里我們?cè)O(shè)置的初始容量為3
}3.3通訊錄的增加需要判斷是否滿(mǎn)了
void AddContact(Contact* con)
{
assert(con); //指針不能為空
if(con->count==con->capicity)
{
(peoInfo*)ptr=(PeoInfo*)realloc(con->data,(con->capicity)*2*sizeof(peoInfo)). //增加人數(shù)前先進(jìn)行判斷是否滿(mǎn)了,如果滿(mǎn)了擴(kuò)大2倍。
if(ptr==NULL)
{
perror("peoInof");
return ;
}
else{
con->data=ptr;
}
}
printf("請(qǐng)輸入名字");
scanf("%s",con->data[con->count].name);
printf("請(qǐng)輸入年齡");
scanf("%d",&(con->data[con->count].age));
printf("請(qǐng)輸入性別");
scanf("%s",con->data[con->count].sex);
printf("請(qǐng)輸入電話");
scanf("%s",con->data[con->count].tele);
printf("請(qǐng)輸入地址");
scanf("%s",con->data[con->count].addr);
con->count++;
printf("增加成功\n");
}^ - ^這樣我們就實(shí)現(xiàn)通訊錄的動(dòng)態(tài)化,就不怕通訊錄人數(shù)會(huì)存滿(mǎn)了。
四、文件的形式存儲(chǔ)通訊錄
前面的通訊錄在程序結(jié)束后發(fā)現(xiàn)人員沒(méi)有辦法進(jìn)行一個(gè)保存,而通過(guò)文件的形式我們可以很好的做到這一點(diǎn),在結(jié)束之前進(jìn)行對(duì)文件的保存,在再一次執(zhí)行程序是輸入保存的文件,但是文件操作也有自己的不足之處,后面用數(shù)據(jù)庫(kù)來(lái)實(shí)現(xiàn)更好,當(dāng)前我們先用文件的形式來(lái)進(jìn)行實(shí)現(xiàn)數(shù)據(jù)的保存,而實(shí)現(xiàn)這個(gè)只需要在結(jié)束時(shí)對(duì)文件保存,在初始化的時(shí)候時(shí)文件信息流入。
4.1人員信息的保存
void SaveContact(Contact*con)
{
assert(con);
//通過(guò)二進(jìn)制寫(xiě)的形式把人員數(shù)據(jù)寫(xiě)入當(dāng)中
FILE* pfWrite=fopen("Contact.txt","wb");
if(pfWrite==NULL)
{
perror("SaveContact");
return ;
}
int i=0;
for(i=0;i<con->count;i++)
{
fwrite(con->data+i,sizeof(peoInfo),1,pfWrite); //一次寫(xiě)一個(gè)人員
}
fclose(pfWrite);
pfWrite=NULL;
}4.2人員信息的流入
void LoadContact(Contact* con)
{
assert(con);
FILE*pfread =fopen("/Users/mamenghao/Desktop/Contact.txt","rb"); //二進(jìn)制來(lái)讀取文件信息
if(pfread==NULL)
{
perror("LoadContact");
return ;
}
peoInfo tmp={0}; //創(chuàng)建一個(gè)人員結(jié)構(gòu)體變量先做臨時(shí)的保存
while( fread(&tmp,sizeof(peoInfo),1,pfread)==1)
{
if(con->count==con->capicity)
{
peoInfo* ptr=(peoInfo*)realloc(con->data,((con->capicity)*2*sizeof(peoInfo)));
if(ptr==NULL)
{
perror("peoInof");
return ;
}
else{
con->data=ptr;
}
} //在增加前先判斷是否容量已經(jīng)滿(mǎn)了
con->data[con->count]=tmp;
con->count++;
}
fclose(pfread);
pfread=NULL;
}總結(jié):
通訊錄本身實(shí)現(xiàn)它并不是特別的難,運(yùn)用的知識(shí)就是順序表的增刪插改功能,而進(jìn)一步的優(yōu)化只需要大家熟練掌握文件的操作和動(dòng)態(tài)內(nèi)存存儲(chǔ)就能很好的實(shí)現(xiàn)啦!!
到此這篇關(guān)于C語(yǔ)言通過(guò)三種方法實(shí)現(xiàn)屬于你的通訊錄的文章就介紹到這了,更多相關(guān)C語(yǔ)言通訊錄內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳解C語(yǔ)言?xún)?nèi)核字符串轉(zhuǎn)換方法
在內(nèi)核開(kāi)發(fā)模式下,初始化字符串也需要調(diào)用專(zhuān)用的初始化函數(shù),如下分別初始化ANSI和UNCODE字符串,本文我們就來(lái)看看代碼是如何實(shí)現(xiàn)的2022-09-09
C語(yǔ)言利用UDP實(shí)現(xiàn)群聊聊天室的示例代碼
UDP是一個(gè)輕量級(jí)、不可靠、面向數(shù)據(jù)報(bào)的、無(wú)連接的傳輸層協(xié)議,多用于可靠性要求不嚴(yán)格,不是非常重要的傳輸,如直播、視頻會(huì)議等等。本文將利用UDP實(shí)現(xiàn)簡(jiǎn)單的群聊聊天室,感興趣的可以了解一下2022-08-08
Qt數(shù)據(jù)庫(kù)相關(guān)應(yīng)用開(kāi)發(fā)總結(jié)
這篇文章主要為大家介紹了在Qt數(shù)據(jù)庫(kù)應(yīng)用開(kāi)發(fā)中的一些經(jīng)驗(yàn)總結(jié),以及一些組件的使用介紹。文中的示例代碼講解詳細(xì),需要的可以參考一下2022-02-02
C語(yǔ)言入門(mén)篇--初識(shí)指針和指針變量
本篇文章是基礎(chǔ)篇,適合c語(yǔ)言剛?cè)腴T(mén)的朋友,本文對(duì)初識(shí)c語(yǔ)言的指針和指針變量做了簡(jiǎn)單的分析,幫助大家快速入門(mén)c語(yǔ)言的世界,更好的理解c語(yǔ)言2021-08-08
深入探索C++中stack和queue的底層實(shí)現(xiàn)
這篇文章主要介紹了C++中的stack和dequeue的底層實(shí)現(xiàn),本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-09-09
C語(yǔ)言編程深入理解取整取余取模問(wèn)題示例分析
這篇文章主要為大家介紹了C語(yǔ)言編程深入理解取整取余取模問(wèn)題的示例分析詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2021-11-11
C++實(shí)現(xiàn)簡(jiǎn)單的計(jì)算器小功能
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)簡(jiǎn)單的計(jì)算器小功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-02-02

