欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C語(yǔ)言實(shí)現(xiàn)可保存的動(dòng)態(tài)通訊錄的示例代碼

 更新時(shí)間:2022年07月21日 14:08:20   作者:蔣靈瑜的流水賬  
這篇文章主要為大家詳細(xì)介紹了如何利用C語(yǔ)言實(shí)現(xiàn)一個(gè)簡(jiǎn)單的可保存的動(dòng)態(tài)通訊錄,文中的示例代碼講解詳細(xì),對(duì)我們學(xué)習(xí)C語(yǔ)言有一定幫助,需要的可以參考一下

一、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)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • C++中vector和map的刪除方法(推薦)

    C++中vector和map的刪除方法(推薦)

    下面小編就為大家?guī)?lái)一篇C++中vector和map的刪除方法(推薦)。小編覺(jué)得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2016-12-12
  • 淺談c++如何實(shí)現(xiàn)并發(fā)中的Barrier

    淺談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-07
  • C語(yǔ)言的多級(jí)指針你了解嗎

    C語(yǔ)言的多級(jí)指針你了解嗎

    這篇文章主要介紹了C語(yǔ)言中的多級(jí)指針,本文給大家介紹的非常詳細(xì),具有參考借鑒價(jià)值,需要的朋友可以參考下,希望能給你帶來(lái)幫助
    2021-08-08
  • C++實(shí)現(xiàn)鄰接表頂點(diǎn)的刪除

    C++實(shí)現(xiàn)鄰接表頂點(diǎn)的刪除

    這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)鄰接表頂點(diǎn)的刪除,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-04-04
  • C++制作《游戲內(nèi)存外掛》詳解

    C++制作《游戲內(nèi)存外掛》詳解

    這篇文章主要介紹了C++制作《游戲內(nèi)存外掛》詳解,文中通過(guò)示例代碼和圖片介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2020-07-07
  • 詳解C語(yǔ)言中的rename()函數(shù)和remove()函數(shù)的使用方法

    詳解C語(yǔ)言中的rename()函數(shù)和remove()函數(shù)的使用方法

    這篇文章主要介紹了詳解C語(yǔ)言中的rename()函數(shù)和remove()函數(shù)的使用方法,是C語(yǔ)言入門(mén)學(xué)習(xí)中的基礎(chǔ)知識(shí),需要的朋友可以參考下
    2015-09-09
  • C/C++左旋字符串實(shí)現(xiàn)代碼舉例

    C/C++左旋字符串實(shí)現(xiàn)代碼舉例

    在C/C++語(yǔ)言中沒(méi)有專(zhuān)門(mén)的字符串變量,通常用字符數(shù)組來(lái)存放字符串,下面這篇文章主要給大家介紹了關(guān)于C/C++左旋字符串實(shí)現(xiàn)的相關(guān)資料,需要的朋友可以參考下
    2023-12-12
  • C語(yǔ)言程序環(huán)境中的預(yù)處理詳解

    C語(yǔ)言程序環(huán)境中的預(yù)處理詳解

    這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言程序環(huán)境中的預(yù)處理,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來(lái)幫助
    2022-02-02
  • C++構(gòu)造和解析Json的使用示例

    C++構(gòu)造和解析Json的使用示例

    今天小編就為大家分享一篇關(guān)于C++構(gòu)造和解析Json的使用示例,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧
    2018-12-12
  • 詳解C語(yǔ)言中動(dòng)態(tài)內(nèi)存管理

    詳解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

最新評(píng)論