C語(yǔ)言實(shí)現(xiàn)可保存的動(dòng)態(tài)通訊錄的示例代碼
一、Contact.h
#pragma once #define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h> #include <assert.h> #include <string.h> #include <stdlib.h> #define MAX_NAME 20 #define MAX_SEX 10 #define MAX_TELE 12 #define MAX_ADDR 30 #define INITIAL 3 typedef struct PeoInfo { char name[MAX_NAME]; char sex[MAX_SEX]; char tele[MAX_TELE]; char addr[MAX_ADDR]; int age; }PeoInfo; typedef struct Contact { PeoInfo* arr;//PeoInfo類(lèi)型的指針,用于指向動(dòng)態(tài)開(kāi)辟的空間 int size;//數(shù)據(jù)個(gè)數(shù) int capacity;//總?cè)萘? }Contact; void ContactInit(Contact* pc);//初始化通訊錄 void ContactPrint(const Contact* pc);//打印 void ContactAdd(Contact* pc);//增加聯(lián)系人信息 void ContactDel(Contact* pc);//刪除聯(lián)系人信息 void ContactFind(const Contact* pc);//查找信息 void ContactModify(Contact* pc);//修改信息 void ContactSortByName(Contact* pc);//排序 void ContactDestroy(Contact* pc);//清空通訊錄 void ContactSave(const Contact* pc);//保存通訊錄為文件
二、Contact.c
1、判斷是否增容
void ContactCapacityIncrease(Contact* pc)//判斷是否增容 { assert(pc); if (pc->size == pc->capacity) { //需要增容 int newcapacity = pc->capacity == 0 ? INITIAL : INITIAL + pc->capacity; PeoInfo* tmp = (PeoInfo*)realloc(pc->arr, newcapacity * sizeof(PeoInfo)); if (tmp == NULL) { printf("%s\n", strerror(errno)); exit(-1); } pc->arr = tmp; pc->capacity = newcapacity; printf("增容成功!\n"); } }
沒(méi)有空間時(shí),為arr開(kāi)辟3個(gè)空間,有空間且空間滿出時(shí),每次為arr擴(kuò)容3個(gè)空間。
2、初始化通訊錄
void ContactInit(Contact* pc)//初始化通訊錄 { assert(pc); pc->size = 0; pc->capacity = 0; pc->arr = NULL; //加載文件的信息到通訊錄 FILE* pf = fopen("Contact.txt", "rb"); if (pf == NULL) { perror("ContactInit:"); exit(-1); } ContactCapacityIncrease(pc); PeoInfo tmp = { 0 }; while (fread(&tmp, sizeof(pc->arr[0]), 1, pf) == 1) { ContactCapacityIncrease(pc); pc->arr[pc->size] = tmp; pc->size++; } fclose(pf); pf = NULL; }
初始化通訊錄并在每次運(yùn)行程序時(shí)把工程目錄下的Contact.txt文本文件加載出來(lái)。實(shí)現(xiàn)通訊錄信息的讀取。
3、打印
void ContactPrint(const Contact* pc)//打印 { assert(pc); printf("姓名\t性別\t電話\t地址\t年齡\n"); for (int i = 0; i < pc->size; i++) { printf("%s\t%s\t%s\t%s\t%d\n", pc->arr[i].name, pc->arr[i].sex, pc->arr[i].tele, pc->arr[i].addr, pc->arr[i].age); } }
for循環(huán)遍歷打印pc->date中的結(jié)構(gòu)體成員
4、增加聯(lián)系人信息
void ContactAdd(Contact* pc)//增加聯(lián)系人信息 { ContactCapacityIncrease(pc); printf("請(qǐng)輸入姓名:\n"); scanf("%s", pc->arr[pc->size].name); printf("請(qǐng)輸入性別:\n"); scanf("%s", pc->arr[pc->size].sex); printf("請(qǐng)輸入電話:\n"); scanf("%s", pc->arr[pc->size].tele); printf("請(qǐng)輸入地址:\n"); scanf("%s", pc->arr[pc->size].addr); printf("請(qǐng)輸入年齡:\n"); scanf("%d", &(pc->arr[pc->size].age)); pc->size++; }
注意增加聯(lián)系人后pc->size++
5、通過(guò)名字查找
static int FindByName(const Contact* pc, const char arr[])//通過(guò)名字查找 { assert(pc && arr); for (int i = 0; i < pc->size; i++) { if (strcmp(pc->arr[i].name, arr) == 0) { return i; } } return -1; }
可以通過(guò)名字查找通訊錄中是否存在聯(lián)系人,存在返回下標(biāo),不存在返回-1
6、刪除聯(lián)系人信息
void ContactDel(Contact* pc)//刪除聯(lián)系人信息 { assert(pc); printf("請(qǐng)輸入姓名查找:"); char arr[20] = { 0 }; scanf("%s", arr); int pos = FindByName(pc, arr);//記錄size的位置 if (pos == -1) { printf("通訊錄沒(méi)有該信息\n"); return; } for (int i = pos; i < pc->size - 1; i++)//移動(dòng)元素 { pc->arr[i] = pc->arr[i + 1]; } pc->size--; printf("刪除成功!\n"); }
通過(guò)靜態(tài)函數(shù)FindByName返回的下標(biāo),通過(guò)for循環(huán)將后續(xù)元素逐個(gè)進(jìn)行覆蓋。
7、查找信息
void ContactFind(const Contact* pc)//查找信息 { assert(pc); printf("請(qǐng)輸入姓名查找:"); char arr[20] = { 0 }; scanf("%s", arr); int pos = FindByName(pc, arr); if (pos != -1) { printf("查找到如下信息:\n"); printf("姓名\t性別\t電話\t地址\t年齡\n"); printf("%s\t%s\t%s\t%s\t%d\n", pc->arr[pos].name, pc->arr[pos].sex, pc->arr[pos].tele, pc->arr[pos].addr, pc->arr[pos].age); } else printf("通訊錄查無(wú)此人!\n"); }
先判斷查找的信息是否在通訊錄中,再打印該下標(biāo)的信息。
8、修改信息
void ContactModify(Contact* pc)//修改信息 { assert(pc); printf("請(qǐng)輸入姓名查找:"); char arr[20] = { 0 }; scanf("%s", arr); int pos = FindByName(pc, arr); if (pos == -1) { printf("找不到\n"); return; } else { printf("請(qǐng)輸入更改后的姓名:\n"); scanf("%s", pc->arr[pos].name); printf("請(qǐng)輸入更改后的性別:\n"); scanf("%s", pc->arr[pos].sex); printf("請(qǐng)輸入更改后的電話:\n"); scanf("%s", pc->arr[pos].tele); printf("請(qǐng)輸入更改后的地址:\n"); scanf("%s", pc->arr[pos].addr); printf("請(qǐng)輸入更改后的年齡:\n"); scanf("%d", &(pc->arr[pos].age)); } }
先判斷查找的信息是否在通訊錄中,再修改該下標(biāo)的信息。
9、排序
int name_cmp(const void* e1, const void* e2) { return strcmp(((PeoInfo*)e1)->name, ((PeoInfo*)e2)->name); } void ContactSortByName(Contact* pc)//排序 { assert(pc && pc->size != 0); qsort(pc->arr, pc->size, sizeof(PeoInfo), name_cmp); printf("排序完成\n"); }
使用qsort函數(shù)排序通訊錄,可參照本文學(xué)習(xí)qsort排序結(jié)構(gòu)體的方法
10、清空通訊錄
void ContactDestroy(Contact* pc)//清空通訊錄 { assert(pc); free(pc->arr); pc->arr = NULL; pc->size = 0; pc->capacity = 0; printf("通訊錄已清空\(chéng)n"); }
free動(dòng)態(tài)開(kāi)辟的空間,將指針置空,pc->size和pc->capacity 置為0
11、保存通訊錄為文件
void ContactSave(const Contact* pc)//保存通訊錄為文件 { assert(pc); FILE* pf = fopen("Contact.txt", "wb"); if (pf == NULL) { perror("ContactSave:"); exit(-1); } for (int i = 0; i < pc->size; i++) { fwrite(pc->arr + i, sizeof(pc->arr[0]), 1, pf); } fclose(pf); pf = NULL; }
將通訊錄中的信息以文件的形式保存起來(lái),下次運(yùn)行可以讀取這部分信息
三、text.c
#include "contact.h" void menu() { printf("###########################\n"); printf("#####1、add 2、del######\n"); printf("#####3、find 4、modify###\n"); printf("#####5、print 6、sort#####\n"); printf("#####7、empty 8、exit#####\n"); printf("###########################\n"); } enum option { ADD = 1, DEL, FIND, MODIFY, PRINT, SORT, EMPTY, EXIT }; int main() { int input = 0; Contact c;//創(chuàng)建一個(gè)通訊錄 ContactInit(&c);//初始化通訊錄 while (1) { menu(); printf("請(qǐng)輸入選項(xiàng):\n"); scanf("%d", &input); if (input == ADD) ContactAdd(&c);//增加聯(lián)系人信息 else if (input == DEL) ContactDel(&c);//刪除聯(lián)系人信息 else if (input == FIND) ContactFind(&c);//查找聯(lián)系人信息 else if (input == MODIFY) ContactModify(&c);//修改聯(lián)系人信息 else if (input == PRINT) ContactPrint(&c);//打印 else if (input == SORT) ContactSortByName(&c);//排序 else if (input == EMPTY) ContactDestroy(&c);//清空通訊錄 else if (input == EXIT) { ContactSave(&c);//保存通訊錄為文件 printf("通訊錄已保存至文件!\n"); ContactDestroy(&c);//清空通訊錄 break;//退出 } else printf("輸入錯(cuò)誤!請(qǐng)重新輸入!\n"); } return 0; }
四、錯(cuò)誤寫(xiě)法分享
圖中綠色波浪線部分有兩個(gè)錯(cuò)誤
1、pc->size在while循環(huán)判斷處,導(dǎo)致最后一次判斷時(shí)(文件已讀取完畢),pc->size也會(huì)++,導(dǎo)致程序運(yùn)行時(shí)越界打印數(shù)據(jù)
2、那么是不是把pc->size++放在while循環(huán)里面就行了呢?不是的,雖然這樣pc->size的大小是正確的,但是當(dāng)通訊錄的容量已滿,下一次循環(huán)進(jìn)來(lái)也是先寫(xiě)入數(shù)據(jù),再進(jìn)行擴(kuò)容,越界訪問(wèn)觸發(fā)斷點(diǎn)。、
正確的寫(xiě)法參照本文初始化通訊錄部分。
五、動(dòng)圖展示
到此這篇關(guān)于C語(yǔ)言實(shí)現(xiàn)可保存的動(dòng)態(tài)通訊錄的示例代碼的文章就介紹到這了,更多相關(guān)C語(yǔ)言動(dòng)態(tài)通訊錄內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C語(yǔ)言實(shí)現(xiàn)動(dòng)態(tài)版通訊錄的代碼分享
- C語(yǔ)言實(shí)現(xiàn)動(dòng)態(tài)版通訊錄的示例代碼
- C語(yǔ)言示例講解動(dòng)態(tài)/文件/靜態(tài)功能版本的通訊錄實(shí)現(xiàn)
- C語(yǔ)言實(shí)現(xiàn)可增容動(dòng)態(tài)通訊錄詳細(xì)過(guò)程
- C語(yǔ)言與C++動(dòng)態(tài)通訊錄超詳細(xì)實(shí)現(xiàn)流程
- C語(yǔ)言動(dòng)態(tài)與靜態(tài)分別實(shí)現(xiàn)通訊錄詳細(xì)過(guò)程
- C語(yǔ)言靜態(tài)動(dòng)態(tài)兩版本通訊錄實(shí)戰(zhàn)源碼
- C語(yǔ)言實(shí)現(xiàn)一個(gè)文件版動(dòng)態(tài)通訊錄流程詳解
相關(guān)文章
淺談c++如何實(shí)現(xiàn)并發(fā)中的Barrier
這篇文章主要介紹了淺談c++如何實(shí)現(xiàn)并發(fā)中的Barrier,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-07-07C++實(shí)現(xiàn)鄰接表頂點(diǎn)的刪除
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)鄰接表頂點(diǎn)的刪除,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-04-04詳解C語(yǔ)言中的rename()函數(shù)和remove()函數(shù)的使用方法
這篇文章主要介紹了詳解C語(yǔ)言中的rename()函數(shù)和remove()函數(shù)的使用方法,是C語(yǔ)言入門(mén)學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下2015-09-09詳解C語(yǔ)言中動(dòng)態(tài)內(nèi)存管理
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言中動(dòng)態(tài)內(nèi)存管理的相關(guān)知識(shí),以及常見(jiàn)的動(dòng)態(tài)內(nèi)存的錯(cuò)誤,文中的示例代碼講解詳細(xì),感興趣的小伙伴可以了解一下2023-07-07