C语言通过三种方法实现属于你的通讯录

   2023-02-09 学习力0
核心提示:目录一、基础版本1.1 通讯录的个人信息(结构体来实现)1.2通讯录名单1.3人员初始化1.4菜单1.5主函数二、功能的实现2.1、增加人数2.2、删除人数2.3、查找2.4、展示2.5、排序(这里我是通过名字)三、通讯录进阶(设置动态存储)3.1通讯录从静态改为动态3.2通

一、基础版本

前提准备: 

1、通讯录里面的人的个人信息(姓名、性别、年龄、手机号、地址)。

2、通讯录名单(来实现人员的增删插改)。

3、人员的初始化。

4、菜单。

1.1 通讯录的个人信息(结构体来实现)

typedef struct peoInfo     //重定义
{
     char name[20]; //姓名
     int age ;        //年龄
     char sex[10];    //性别
     char tele[12];    //手机号
     char addr[30];     //地址
 }peoInfo;

1.2通讯录名单

typedef struct Contact
{
    peoInfo data[100]; //人员名单 结构体数组
    int count;         //人员数量
}Contact;

1.3人员初始化

void InitContact(Contact* con)
{
    assert(con);
    con->count=0; //初始人数为0
    memset(con->data,0,sizeof(con->data)); //这是一个c语言的库函数把数组里面的数全部初始化为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主函数

enum a
{
    ext, //这里用了简单的枚举的方法 枚举的初始值为0
    add,
    del,
    serch,
    modify,
    show,
    sort
};
int main()
{
    int choice; 
    Contact con; //通讯录
    do
    {
        menu();
        printf("请选择功能\n");
        scanf("%d",&choice);
        switch(choice)
        {
            case add:
                AddContact(&con); //实现增加功能的函数
                break ;
            case del:
                DelContanct(&con); //实现删除功能的函数
                break ;
            case serch:
                SerContact(&con); //实现查找功能的函数
                break ;
            case modify:            
                ModContact(& con); //实现修改功能的函数
                break ;
            case show:
                ShowContact(&con); //实现展示功能的函数
                break ;
            case sort:
                SorContact(&con);     //实现排序功能的函数
                break ;
            case ext :
                printf("程序结束\n");
                break;
            default :
                printf("输入有误请重新输入\n");
        }
    }while(choice);
    return 0;
    
}

二、功能的实现

2.1、增加人数

void AddContact(Contact* con)
{
    assert(con); //指针不能为空
    if(con->count==Max)
    {
        printf("存放已满\n");
        return ;
    }
    printf("请输入名字");
    scanf("%s",con->data[con->count].name);
    printf("请输入年龄");
    scanf("%d",&(con->data[con->count].age));
    printf("请输入性别");
    scanf("%s",con->data[con->count].sex);
    printf("请输入电话");
    scanf("%s",con->data[con->count].tele);
    printf("请输入地址");
    scanf("%s",con->data[con->count].addr);
    con->count++;
    printf("增加成功\n");
    
}

2.2、删除人数

 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;
         }
    else 
        {
            return -1;
        }
     }
 void DelContanct(Contact* con)
{
    char name[20]={0};
    printf("请输入你想删除的名字\n");
    assert(con);
    int i;
    scanf("%s",name); 
    int pos=xiaomafind(con,name);
    if(pos==-1)
    {
        printf("没有你要删除的人\n");
    }
    else
    {
        for(i=pos;i<con->count;i++)
        {
            con->data[i]=con->data[i+1]; //通过数组后一个覆盖前一个来完成删除功能
        }
        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("请输入你想查找的名字\n");
    assert(con);
    scanf("%s",name);
    int find =xiaomafind(con,name);
    if(find==-1)
    {
        printf("没有你要查找的用户\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、排序(这里我是通过名字)

^ _ ^用qsort 函数发现很很容的都能进行排序,我们只需要告诉他我们的排序方法是什么就可以很容易的进行一下排序

int cmp_stu_by_name(const void*e1,const void*e2)
{
    return strcmp(((peoInfo*)e1)->name,((peoInfo*)e2)->name); 
    } //如不不理解的看我指针讲解中回调函数
void SorContact(Contact* con)
{
    assert(con);
    
    qsort(con->data, con->count, sizeof(peoInfo), cmp_stu_by_name); //这里运用了一个c语言的库函数qsort函数,在指针进阶中我也讲述了如何使用,如果不会的可以看一下指针的进阶。
}

三、通讯录进阶(设置动态存储)

动态版本只需要改三个位置就行啦

3.1通讯录从静态改为动态

//静态版本 
//typedef struct Contact
//{
 //   peoInfo data[100]; //人员名单 结构体数组
  //  int count;         //人员数量
//}Contact;

//动态版本
​​​​​​​typedef struct Contact
{
    peoInfo *data; //人员名单 结构体指针
    int count;         //人员数量
    int capicity;  //通讯录容量
}Contact;

3.2通讯录的初始化

 //静态版本
//void InitContact(Contact* con)
//{
  //  assert(con);
  //  con->count=0; //初始人数为0
 //   memset(con->data,0,sizeof(con->data)); //这是一个c语言的库函数把数组里面的数全部初始化为0    
//}
  //动态版本
void  InitContact(Contact* con)
{
    assert(con);
    con->count=0;
    con->data=(PeoInf0*)calloc(3,sizeof(peoInfo*)); 
    if(con->data=NULL)
    {
        perrror("con->data"); //如果空指针打印错误原因
    }
    con->capicity=3; //这里我们设置的初始容量为3
  
}

3.3通讯录的增加需要判断是否满了

void AddContact(Contact* con)
{
    assert(con); //指针不能为空
    if(con->count==con->capicity)
    {
        (peoInfo*)ptr=(PeoInfo*)realloc(con->data,(con->capicity)*2*sizeof(peoInfo)). //增加人数前先进行判断是否满了,如果满了扩大2倍。
    if(ptr==NULL)
        {
            perror("peoInof");
            return ;
        }
        else{
            con->data=ptr;
        }
    }
    printf("请输入名字");
    scanf("%s",con->data[con->count].name);
    printf("请输入年龄");
    scanf("%d",&(con->data[con->count].age));
    printf("请输入性别");
    scanf("%s",con->data[con->count].sex);
    printf("请输入电话");
    scanf("%s",con->data[con->count].tele);
    printf("请输入地址");
    scanf("%s",con->data[con->count].addr);
    con->count++;
    printf("增加成功\n");
    
}

^ - ^这样我们就实现通讯录的动态化,就不怕通讯录人数会存满了。

四、文件的形式存储通讯录

前面的通讯录在程序结束后发现人员没有办法进行一个保存,而通过文件的形式我们可以很好的做到这一点,在结束之前进行对文件的保存,在再一次执行程序是输入保存的文件,但是文件操作也有自己的不足之处,后面用数据库来实现更好,当前我们先用文件的形式来进行实现数据的保存,而实现这个只需要在结束时对文件保存,在初始化的时候时文件信息流入。

4.1人员信息的保存

void SaveContact(Contact*con)
{
    assert(con);
     //通过二进制写的形式把人员数据写入当中
    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); //一次写一个人员
    }
    fclose(pfWrite);
    pfWrite=NULL;
}

4.2人员信息的流入

void LoadContact(Contact* con)
{
    assert(con);
    FILE*pfread =fopen("/Users/mamenghao/Desktop/Contact.txt","rb"); //二进制来读取文件信息
    if(pfread==NULL)
    {
        perror("LoadContact");
        return ;
    }
    peoInfo tmp={0}; //创建一个人员结构体变量先做临时的保存
   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;
           }
       } //在增加前先判断是否容量已经满了
 
       con->data[con->count]=tmp;
       con->count++;
   }
    fclose(pfread);
    pfread=NULL;
    
}

总结:

通讯录本身实现它并不是特别的难,运用的知识就是顺序表的增删插改功能,而进一步的优化只需要大家熟练掌握文件的操作和动态内存存储就能很好的实现啦!!

原文地址:https://blog.csdn.net/m0_63177573/article/details/127858400
 
标签: C语言 通讯录
反对 0举报 0 评论 0
 

免责声明:本文仅代表作者个人观点,与乐学笔记(本网)无关。其原创性以及文中陈述文字和内容未经本站证实,对本文以及其中全部或者部分内容、文字的真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
    本网站有部分内容均转载自其它媒体,转载目的在于传递更多信息,并不代表本网赞同其观点和对其真实性负责,若因作品内容、知识产权、版权和其他问题,请及时提供相关证明等材料并与我们留言联系,本网站将在规定时间内给予删除等相关处理.

  • Rust应用调用C语言动态库的操作方法
    目录外部功能接口FFIUDP套接字的读超时Rust调用C语言动态库中的函数避免重复造***,使用Rust官方C语言库外部功能接口FFI虽然高级(脚本)编程语言的功能丰富,表达能力强,但对底层的一些特殊操作的支持并不完善,就需要以其他编程语言来实现。调用其他编程语
  • Delphi中获取Unix时间戳及注意事项(c语言中tim
    uses DateUtils;DateTimeToUnix(Now) 可以转换到unix时间,但是注意的是,它得到的时间比c语言中time()得到的时间大了8*60*60这是因为Now是当前时区的时间,c语言中time()是按格林威治时间计算的,北京时间比格林威治时间多了8小时DateTimeToUnix(Now)-8*60*
    02-09
  • Unicode与UTF-8互转(c语言和lua语言) python
    1. 基础1.1 ASCII码我们知道, 在计算机内部, 全部的信息终于都表示为一个二进制的字符串. 每个二进制位(bit)有0和1两种状态, 因此八个二进制位就能够组合出 256种状态, 这被称为一个字节(byte). 也就是说, 一个字节一共能够用来表示256种不同的状态, 每个状态
    02-09
  • R语言中cat函数 c语言cat命令
    R语言中cat函数 c语言cat命令
    R语言中cat函数。1、测试1cat("aa","bb")cat("aa","bb",sep = "_")  2、测试2a = 100b = 300c = "abcd"cat(a,b,c)cat(a,b,c,sep = "_") 3、测试3a = c("aaa", "bbb", "ccc")b = 1:4ca
    02-09
  • R语言之merge详解 c语言merge函数代码
    merge是R语言中用来合并数据框的函数merge函数的声明:?1234merge(x, y, by = intersect(names(x), names(y)),      by.x = by, by.y = by, all = FALSE, all.x = all, all.y = all,      sort = TRUE, suffixes = c(".x"
    02-09
  • R语言调用的C语言源代码查询 R语言 c
    R语言使用时可以调用自己写的C代码,但是有些C函数是软件包自带的,怎么查询在使用软件包 kerfdr 时,涉及到一个函数y = .C("massdist", x = as.double(xtrunc), xmass = as.double(tau[trunc]/sum(tau[trunc])), nx = nx, xlo = as.double(lo), xhi = as.dou
    02-09
  • centos安装与配置R语言 centos配置c语言环境
    Linux下安装R语言一、编译安装      由于采用编译安装,所以需要用到gcc编译环境,在编译前check文件时还会用到libXt-devel和readline-devel两个依赖,所以在编译R语言源码时先将这些工具和依赖包准备好。readline-devel 也可以不安装,不安装此包R语言编
    02-09
  • C语言利用链表实现学生成绩管理系统
    链表是一种常见的基础数据结构,结构体指针在这里得到了充分的利用。链表可以动态的进行存储分配,也就是说,链表是一个功能极为强大的数组,他可以在节点中定义多种数据类型,还可以根据需要随意增添,删除,插入节点。链表都有一个头指针,一般以head来表示
  • C++集体数据交换实现示例讲解 c语言两个数据交
    目录一、说明二、示例和代码一、说明到目前为止介绍的功能共享一对一的关系:即一个进程发送和一个进程接收。链接是通过标签建立的。本节介绍在多个进程中调用相同参数但执行不同操作的函数。对于一个进程,函数可能会发送数据,对于另一个进程,它可能会接收
  • C语言中#if的使用 c语言中%c
    目录#if定义#if使用#if的后面接的是表达式#if defined的使用ifdef的使用结尾#if定义#if和#endif是一组同时使用的,叫做条件编译指令。#if与#define、#include等指令一样是由预处理器这个强大的工具处理的,预处理器可以在编译前处理c程序。#if使用#if的后面接
    02-09 C语言#if
点击排行