c语言实现通讯录(用三种方法来实现一个属于你的通讯录)

news2025/7/9 5:52:58

前沿:


  ~~在一个周前,我用c语言实现了一个银行账号的功能,在总结当中我说了要实现一个通讯录,来实现人员的增删插改功能,而现在就是我实现它的时候!!,本文呢小马将通过三种形式来实现用户的增删插改,其实也就是一点点的优化版本,从静态的实现,到动态的实现,最后以文件的形式来完成,请大家和我一起往下看吧!➡︎ 

 思维导图:↘️      ⤵️

↘️

目录

一、基础版本:

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

1..2通讯录名单:

1.3人员初始化:

1.4菜单:

1.5主函数:

二、功能的实现:

2.1、增加人数:

2.2、删除人数:

2.3、查找:

2.4、展示:

2.5、排序(这里我是通过名字):

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

3.1通讯录从静态改为动态:

3.2通讯录的初始化:

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

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

4.1人员信息的保存:

4.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;
    
}

总结:

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

最后小马码文不易,觉得有帮助的话请多多支持哈^ _ ^

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/9129.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于python命令流及代码的Plaxis自动化建模

有限单元法在岩土工程问题中应用非常广泛&#xff0c;很多商业软件如Plaxis/Abaqus/Comsol等都采用有限单元解法。在使用各大软件进行数值模拟建模的过程中&#xff0c;您是否发现GUI界面中重复性的点击输入工作太繁琐&#xff1f;从而拖慢了设计或方案必选进程&#xff1f; 本…

SpringBoot八种bean的加载方式一学就会

目录 文章目录[toc]第一种bean的加载方式-配置文件第二种加载bean方式-注解和扫描创建第三方的bean对象第三种加载bean方式-不使用配置文件扩展-bean的加载方式扩展FactoryBean<>扩展-ImportResource导入配置文件扩展-proxyBeanMethods属性-产生代理对象第四种加载bean方…

一文读懂,python实现常用的数据编码和对称加密

相信很多使用 python 的小伙伴在工作中都遇到过&#xff0c;对数据进行相关编码或加密的需求&#xff0c;今天这篇文章主要给大家介绍对于一些常用的数据编码和数据加密的方式&#xff0c;如何使用 python 去实现。话不多说&#xff0c;接下来直接进入主题&#xff1a; 1、bas…

Windows OpenGL ES 图像灰度图

目录 一.OpenGL ES 图像灰度图 1.原始图片2.效果演示 二.OpenGL ES 图像灰度图源码下载三.猜你喜欢 零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 基础 零基础 OpenGL ES 学习路线推荐 : OpenGL ES 学习目录 >> OpenGL ES 特效 零基础 Open…

Redis添加至windows服务

1.进入redis文件夹 打开redis所在文件夹&#xff0c;在资源管理器地址栏输入cmd&#xff0c;打开管理员命令提示符窗口。 2.配置服务参数 打开redis.windows.conf文件&#xff0c;找到“requirepass”配置项&#xff0c;此处设置redis密码&#xff0c;如果不需要密码&#xff0…

Python如何使用HanNLP工具

Python如何使用HanNLP工具目录系统安装配置 JDK 1.8系统安装 Visual C 2015下载 HanNLP 包测试HanNLP工具目录 目标&#xff1a;使用pycharm调用HanNLP工具完成对文本的分词、自动摘要、关键词提取等任务。 系统安装配置 JDK 1.8 1、windows环境下载 JDK 1.8 2、安装 JDK 1.…

Oracle PrimaveraUnifier空间管理器(Space Manager)

目录 前言 介绍 前言 在Oracle Primavera Unifier设施和资产管理基础产品中除了业务流程及配置管理器之外&#xff0c;其预配置设计还包含本文要介绍的空间管理器&#xff0c;即Space Manager 在Unifier中&#xff0c;空间管理器是用户可以执行设施管理任务的地方。空间管…

xv6---Lab1: Xv6 and Unix utilities

目录 参考资料&#xff1a; 1.1进程和内存 1.2 I/O 和文件描述符 1.3管道 源码&#xff1a; 调试环境搭建 sleep PingPong primes find xargs 参考资料&#xff1a; Lab: Xv6 and Unix utilities xv6-book翻译(自用&#xff09;第一章 - 知乎 1.1进程和内存 一个xv…

html中css的基础学习

小李胖了吗 I 都说秋天适合思念&#xff0c;其实更适合见面【小李胖了吗 I 都说秋天适合思念&#xff0c;其实更适合见面】 https://www.bilibili.com/video/BV19g411B7uL/?share_sourcecopy_web&vd_source385ba0043075be7c24c4aeb4aaa73352 通过本博文的学习&#xff0c…

常见的软件测试面试题,千万别答错了

软件测试的童鞋们&#xff0c;在面试测试工作时&#xff0c;一定遇到面试官问过这个问题&#xff1a; 软件测试的目的意义是什么&#xff1f;大家是怎么回答的呢&#xff1f;如果这个问题回答好了&#xff0c;说明你对软件测试工作的价值与意义了如指掌。 有经验的测试人员可…

[数据结构]栈和队列面试题解析

作者&#xff1a; 华丞臧. 专栏&#xff1a;【数据结构】 各位读者老爷如果觉得博主写的不错&#xff0c;请诸位多多支持(点赞收藏关注)。如果有错误的地方&#xff0c;欢迎在评论区指出。 推荐一款刷题网站 &#x1f449; LeetCode刷题网站 文章目录一、有效括号题目描述解题思…

freemarker+yml介绍 以及freemarker与JSP的区别

目录 1. freemarker介绍 2. freemarker使用步骤 2.1 在pom.xml引入freeMarker的依赖包 2.2 在springboot中添加freemarker配置 2.3 编写模板文件*.ftl(当做jsp使用即可) ​编辑 2.4 访问控制器后进行页面跳转 3. freemarker常用语法 3.1 取值 3.2 条件 3.3 循环 3…

超赞:不愧是“阿里内部Redis学习笔记”从头到尾,全是精华

近几年&#xff0c;随着移动互联网的飞速发展&#xff0c;我们享受着整个社会的技术进步带来的便利&#xff0c;但同时也给从业者带来了如何保证项目的高并发、低延时的技术挑战&#xff0c;相应的互联网技术也随之发生了重大变革&#xff0c;NoSQL技术得到了蓬勃的发展。 Red…

【JavaSE】抽象类与接口

文章目录抽象类的概念抽象类的语法抽象类的特性接口的概念接口的语法接口的特性多接口的实现接口的继承抽象类的概念 什么是抽象类呢&#xff1f;我们先来看一个例子&#xff1a;一个父类是动物类&#xff0c;两个子类一个狗狗类&#xff0c;一个猫猫类。 Animal中有一个dark…

有趣的statement stack

引子 在使用events_statements_current的过程中发现&#xff0c;同一线程在同一时刻&#xff0c;可能有多条记录&#xff0c;与直观感觉不太一样&#xff0c;于是跟踪了一下内部实现&#xff0c;有了本文。 STATEMENT STACK的定义 STATEMENT STACK 是events_statements_curr…

Linux基础内容(11)—— 进程理解

目录 1.进程状态 1.只针对操作系统的宏观概念 2.Linux的进程状态 1.运行状态(R) 2.休眠状态(S) 3.暂停状态/浅度睡眠状态(T) 4.深度睡眠状态(D) 5.当前进程正在被追踪(t) 6.死亡状态(X) 7.僵尸状态(Z) 8.孤儿进程 2.进程的优先级 1.优先级定义 2.Linux的优先级表现…

matlab 小数据法求liyapunov指数

1、内容简介 略 625-可以交流、咨询、答疑 2、内容说明 摘 要&#xff1a;从 Lyapunov 指数的定义出发&#xff1a;在常用计算最大 Lyapunov 指数的基础上&#xff0c;将自相关法和 G-P 法应用于小数 据量法中&#xff0c;得到了一种计算最大 Lyapunov 指数的改进小数据量法…

推特的算法规则你知道多少?

如果玩Twitter却不明白Twitter算法是如何运作的&#xff0c;就如同贸然出征却忘了带上武器。 社交媒体的算法正在成为你在平台中所见内容的核心&#xff0c;当知晓了该平台的推荐算法&#xff0c;自然也就知道发布什么样的内容有助于被更多用户看到。因此&#xff0c;了解了算…

Spring之依赖注入

文章目录前言一、set注入1.1 UserDao类&#xff1a;1.2 UserService类&#xff1a;1.3 spring.xml配置文件&#xff1a;1.4 测试类&#xff1a;1.5 执行结果&#xff1a;二、构造方法注入2.1 UserDao2.2 UserService类&#xff1a;1.3 spring.xml配置文件&#xff1a;2.4 测试类…

微服务改造过程中那些必须重视的问题

“微服务”近几年尤其火热&#xff0c;各大厂都在进行微服务化改造和微服务建设&#xff0c;想享受微服务化带来的好处以便对自己的系统进行改造。分布式实验室特约记者李鹏采访了广州轻阅科技系统架构师陈珙&#xff0c;就微服务与SOA的区别与联系、企业引入微服务会带来的问题…