c语言:通讯录管理系统(动态分配内存版)

        前言:在大多数高校内,都是通过设计一个通讯录管理系统来作为c语言课程设计,通过一个具体的系统设计将我们学习过的结构体和函数等知识糅合起来,可以很好的锻炼学生的编程思维,本文旨在为通讯录管理系统的设计提供思路和示例讲解,并且对如何动态的分配内存空间使得程序更高效的运行进行了讲解

  

        本文是在基础的通讯录管理系统上进行改进(文末有完整代码,欢迎大家使用),如果对于基础的通讯录管理系统有什么不懂的地方,可以参考笔者的上一篇文章,里面有详细的万字解读:

c语言:通讯录管理系统(增删查改)

目录

一.基础静态版本 (改进前)

头文件部分

函数实现部分

主函数部分

二.结构体的更改

三.扩容的设计

四.释放空间

五.最终完整代码 (改进后)

头文件部分

函数的实现部分 

主函数部分


一.基础静态版本 (改进前)

这里我们给出基础版本的代码,然后本文后续内容都是基于此进行改进

我们分为 3 个文件来设计:

  • Contact.h: 包含头文件的声明,对函数的声明,以及宏的申明
  • Contact.cpp: 通讯录管理系统中具体每一个函数的实现
  • test.cpp: 主函数,根据用户的选择进行调用相应的函数

头文件部分(Contact.h

#pragma once
#include<stdio.h>
#include<assert.h>
#include<string.h>#define Name_Max 20
#define Tel_Number 12
#define Sex_Max 5
#define Address_Max 30
#define Contact_Max 100//联系人结构体
typedef struct PeopleInformation
{char name[Name_Max];char telnumber[Tel_Number];int age;char sex[Sex_Max];char address[Address_Max];
}PeoInfor;//通讯录结构体
typedef struct Contact
{PeoInfor data[Contact_Max];//结构体数组存放联系人结构体int size;//记录当前通讯录中有多少个联系人
}Contact;//目录
void menu();//初始化通讯录
void InitContact(Contact* cp);//增加联系人
void AddContact(Contact* cp);//删除联系人
void DelContact(Contact* cp);//通过姓名进行查找联系人
int FindPeople(Contact* cp, char name[]);//展示全部通讯录信息
void ShowContact(const Contact* cp);//查询联系人
void SeachPeople(Contact* cp);//修改联系人信息
void ModifyContact(Contact* cp);

函数实现部分(Contact.cpp

#define _CRT_SECURE_NO_WARNINGS 1
#include "Contact.h"void menu()
{printf("\n");printf("-----------------------------\n");printf("---   1.添加联系人      -----\n");printf("---   2.删除联系人      -----\n");printf("---   3.查找联系人      -----\n");printf("---   4.修改联系人信息  -----\n");printf("---   5.显示全部信息    -----\n");printf("---   0.退出通讯录      -----\n");printf("-----------------------------\n");
}//初始化通讯录
void InitContact(Contact* cp)
{//判断非空assert(cp);cp->size = 0;memset(cp->data, 0, sizeof(cp->data));
}//增加联系人
void AddContact(Contact* cp)
{//判断非空assert(cp);//判断未满if (cp->size == Contact_Max){printf("通讯录已满,无法再添加新的联系人\n");return;}printf("请输入要添加的联系人的姓名:\n");scanf("%s", cp->data[cp->size].name);printf("请输入要添加的联系人的电话号:\n");scanf("%s", cp->data[cp->size].telnumber);printf("请输入要添加的联系人的年龄:\n");scanf("%d", &(cp->data[cp->size].age));printf("请输入要添加的联系人的性别:\n");scanf("%s", cp->data[cp->size].sex);printf("请输入要添加的联系人的住址:\n");scanf("%s", cp->data[cp->size].address);cp->size++;printf("添加成功\n");
}//通过姓名进行查找联系人
int FindPeople(Contact* cp, char name[])
{assert(cp);for (int i = 0; i < cp->size; i++){if (strcmp(cp->data[i].name, name) == 0){return i;}}return -1;
}//删除联系人
void DelContact(Contact* cp)
{assert(cp);char name[Name_Max];if (cp->size == 0){printf("通讯录为空,无需删除\n");return;}printf("请输入选择删除的联系人的姓名:\n");scanf("%s", name);int ret = FindPeople(cp, name);if (ret == -1){printf("要删除的联系人不存在\n");return;}for (int i = ret; i < cp->size-1 ; i++){cp->data[i] = cp->data[i + 1];}cp->size--;printf("删除成功\n");
}//查询联系人
void SeachPeople(Contact* cp)
{assert(cp);char name[Name_Max];if (cp->size == 0){printf("通讯录为空\n");return;}printf("请输入选择查找的联系人的姓名:\n");scanf("%s", name);int ret = FindPeople(cp, name);if (ret == -1){printf("要查找的联系人不存在\n");return;}//名字  年龄  性别    电话    地址//xxx   xxx    xxx    xxx     xxxprintf("%-10s%-5s%-5s%-12s%-30s\n", "名字", "年龄", "性别", "电话", "地址");//打印个人的信息printf("%-10s%-5d%-5s%-12s%-30s\n", cp->data[ret].name, cp->data[ret].age, cp->data[ret].sex, cp->data[ret].telnumber, cp->data[ret].address);
}//展示全部通讯录信息
void ShowContact(const Contact* cp)
{assert(cp);if (cp->size == 0){printf("通讯录为空\n");return;}//名字  年龄  性别    电话    地址//xxx   xxx    xxx    xxx     xxxprintf("%-10s%-5s%-5s%-12s%-30s\n", "名字", "年龄", "性别", "电话", "地址");for (int i = 0; i < cp->size; i++){//打印每个人的信息printf("%-10s%-5d%-5s%-12s%-30s\n",cp->data[i].name, cp->data[i].age, cp->data[i].sex, cp->data[i].telnumber, cp->data[i].address);}
}//修改联系人信息
void ModifyContact(Contact* cp)
{assert(cp);char name[Name_Max];if (cp->size == 0){printf("通讯录为空\n");return;}printf("请输入选择修改的联系人的姓名:\n");scanf("%s", name);int ret = FindPeople(cp, name);if (ret == -1){printf("要修改的联系人信息不存在\n");return;}printf("请输入要修改的联系人的姓名:\n");scanf("%s", cp->data[ret].name);printf("请输入要修改的联系人的电话号:\n");scanf("%s", cp->data[ret].telnumber);printf("请输入要修改的联系人的年龄:\n");scanf("%d", &(cp->data[ret].age));printf("请输入要修改的联系人的性别:\n");scanf("%s", cp->data[ret].sex);printf("请输入要修改的联系人的住址:\n");scanf("%s", cp->data[ret].address);printf("修改成功\n");
}

主函数部分(test.cpp

#define _CRT_SECURE_NO_WARNINGS 1
#include "Contact.h"//枚举,增加程序的可读性
enum options
{EXIT,ADD,DEL,SEACH,MODIFY,SHOW
};int main()
{int input = 0;//创建通讯录Contact con;//初始化通讯录InitContact(&con);do{menu();printf("请输入你的选择: ");scanf("%d", &input);switch (input){//增加联系人信息case ADD:AddContact(&con);break;//删除联系人信息case DEL:DelContact(&con);break;//查找某个联系人的信息case SEACH:SeachPeople(&con);break;//修改某个联系人的信息case MODIFY:ModifyContact(&con);break;//展示通讯录内的每一个联系人的信息case SHOW:ShowContact(&con);break;//退出通讯录管理系统case EXIT:printf("通讯录已退出\n");break;//预防非法输入default:printf("输入错误,请重新输入\n");break;}}while(input);return 0;
}

二.结构体的更改

        动态的分配内存就意味着通讯录这个结构体要动态的分配内存,根据通讯录内的信息进行分配,所以我们在这里对于通讯录结构体进行更改

//通讯录结构体
typedef struct Contact
{PeoInfor* data;//结构体数组存放联系人结构体int size;//记录当前通讯录中有多少个联系人int capacity;//记录当前存放的容量
}Contact;
  •         在这里我们将 data 从一个结构体数组改成了结构体指针,然后后续再使用 这个指针指向我们动态开辟的内存就完成了我们的需求设计
  •         并且新增了个变量 capacity 用来记录当前通讯录内的最大容量,当联系人的数量和容量相同的时候,也就是通讯录满容的时候,我们再使用 realloc 重新分配新的内存空间

三.扩容的设计

        我们封装一个函数方便我们添加新的联系人的时候进行扩容,先判断当前通讯录是否已满,如果满了就进行扩容,每一次扩容扩展 2 个联系人结构体的大小

  •         首先是判断部分,当当前通讯录的容量等于通讯录内实际存放的数据的大小的时候,我们就判定为通讯录已满,然后我们使用 realloc 开辟新的空间,比之前大 2 个联系人结构体的大小
  •         为了程序的健全性,我们也要判断开辟空间是否成功,如果成功就通过 ptr指针 指向联系人的数据部分,用 data 接收,如果开辟失败,我们就打印报错信息
void CheckContact(Contact* cp)
{if (cp->size == cp->capacity){PeoInfor* ptr = (PeoInfor*)realloc(cp->data, (cp->capacity + 2) * sizeof(PeoInfor));if (ptr != NULL){cp->data = ptr;cp->capacity += 2;printf("增容成功\n");}else{perror("AddContact->realloc");return;}}
}

四.释放空间

        由程序员申请开辟的空间也应当由程序员设置进行释放,在这个通讯录管理系统中也是如此,我们需要找到合适的释放位置,也就是当用户退出通讯录的时候,我们手动进行对开辟的空间进行释放,以避免造成内存泄漏

        那我们这里就封装一个释放空间的函数:

//销毁通讯录
void DestoryContact(Contact* cp)
{free(cp->data);cp->data = NULL;cp->size = 0;cp->capacity = 0;
}

五.最终完整代码 (改进后)

头文件部分(Contact.h:

#pragma once
#pragma once
#include<stdio.h>
#include<assert.h>
#include<string.h>
#include<stdlib.h>#define Name_Max 20
#define Tel_Number 12
#define Sex_Max 5
#define Address_Max 30
#define Contact_Max 100
#define Contact_SZ 3//联系人结构体
typedef struct PeopleInformation
{char name[Name_Max];char telnumber[Tel_Number];int age;char sex[Sex_Max];char address[Address_Max];
}PeoInfor;//通讯录结构体
typedef struct Contact
{PeoInfor* data;//结构体数组存放联系人结构体int size;//记录当前通讯录中有多少个联系人int capacity;//记录当前存放的容量
}Contact;//目录
void menu();//初始化通讯录
void InitContact(Contact* cp);//增加联系人
void AddContact(Contact* cp);//删除联系人
void DelContact(Contact* cp);//通过姓名进行查找联系人
int FindPeople(Contact* cp, char name[]);//展示全部通讯录信息
void ShowContact(const Contact* cp);//查询联系人
void SeachPeople(Contact* cp);//修改联系人信息
void ModifyContact(Contact* cp);//扩容
void CheckContact(Contact* cp);//销毁通讯录
void DestoryContact(Contact* cp);

函数的实现部分 (Contact.cpp

#define _CRT_SECURE_NO_WARNINGS 1
#include "Contact.h"void menu()
{printf("\n");printf("-----------------------------\n");printf("---   1.添加联系人      -----\n");printf("---   2.删除联系人      -----\n");printf("---   3.查找联系人      -----\n");printf("---   4.修改联系人信息  -----\n");printf("---   5.显示全部信息    -----\n");printf("---   0.退出通讯录      -----\n");printf("-----------------------------\n");
}//初始化通讯录
void InitContact(Contact* cp)
{//判断非空assert(cp);cp->size = 0;cp->capacity = Contact_SZ;cp->data =(PeoInfor*)calloc(cp->capacity, sizeof(PeoInfor));if (cp->data == NULL){perror("InitContact->calloc");return;}
}void CheckContact(Contact* cp)
{if (cp->size == cp->capacity){PeoInfor* ptr = (PeoInfor*)realloc(cp->data, (cp->capacity + 2) * sizeof(PeoInfor));if (ptr != NULL){cp->data = ptr;cp->capacity += 2;printf("增容成功\n");}else{perror("AddContact->realloc");return;}}
}//增加联系人
void AddContact(Contact* cp)
{//判断非空assert(cp);//判断满后扩容CheckContact(cp);printf("请输入要添加的联系人的姓名:\n");scanf("%s", cp->data[cp->size].name);printf("请输入要添加的联系人的电话号:\n");scanf("%s", cp->data[cp->size].telnumber);printf("请输入要添加的联系人的年龄:\n");scanf("%d", &(cp->data[cp->size].age));printf("请输入要添加的联系人的性别:\n");scanf("%s", cp->data[cp->size].sex);printf("请输入要添加的联系人的住址:\n");scanf("%s", cp->data[cp->size].address);cp->size++;printf("添加成功\n");
}//通过姓名进行查找联系人
int FindPeople(Contact* cp, char name[])
{assert(cp);for (int i = 0; i < cp->size; i++){if (strcmp(cp->data[i].name, name) == 0){return i;}}return -1;
}//删除联系人
void DelContact(Contact* cp)
{assert(cp);char name[Name_Max];if (cp->size == 0){printf("通讯录为空,无需删除\n");return;}printf("请输入选择删除的联系人的姓名:\n");scanf("%s", name);int ret = FindPeople(cp, name);if (ret == -1){printf("要删除的联系人不存在\n");return;}for (int i = ret; i < cp->size - 1; i++){cp->data[i] = cp->data[i + 1];}cp->size--;printf("删除成功\n");
}//查询联系人
void SeachPeople(Contact* cp)
{assert(cp);char name[Name_Max];if (cp->size == 0){printf("通讯录为空\n");return;}printf("请输入选择查找的联系人的姓名:\n");scanf("%s", name);int ret = FindPeople(cp, name);if (ret == -1){printf("要查找的联系人不存在\n");return;}//名字  年龄  性别    电话    地址//xxx   xxx    xxx    xxx     xxxprintf("%-10s%-5s%-5s%-12s%-30s\n", "名字", "年龄", "性别", "电话", "地址");//打印个人的信息printf("%-10s%-5d%-5s%-12s%-30s\n", cp->data[ret].name, cp->data[ret].age, cp->data[ret].sex, cp->data[ret].telnumber, cp->data[ret].address);
}//展示全部通讯录信息
void ShowContact(const Contact* cp)
{assert(cp);if (cp->size == 0){printf("通讯录为空\n");return;}//名字  年龄  性别    电话    地址//xxx   xxx    xxx    xxx     xxxprintf("%-10s%-5s%-5s%-12s%-30s\n", "名字", "年龄", "性别", "电话", "地址");for (int i = 0; i < cp->size; i++){//打印每个人的信息printf("%-10s%-5d%-5s%-12s%-30s\n", cp->data[i].name, cp->data[i].age, cp->data[i].sex, cp->data[i].telnumber, cp->data[i].address);}
}//修改联系人信息
void ModifyContact(Contact* cp)
{assert(cp);char name[Name_Max];if (cp->size == 0){printf("通讯录为空\n");return;}printf("请输入选择修改的联系人的姓名:\n");scanf("%s", name);int ret = FindPeople(cp, name);if (ret == -1){printf("要修改的联系人信息不存在\n");return;}printf("请输入要修改的联系人的姓名:\n");scanf("%s", cp->data[ret].name);printf("请输入要修改的联系人的电话号:\n");scanf("%s", cp->data[ret].telnumber);printf("请输入要修改的联系人的年龄:\n");scanf("%d", &(cp->data[ret].age));printf("请输入要修改的联系人的性别:\n");scanf("%s", cp->data[ret].sex);printf("请输入要修改的联系人的住址:\n");scanf("%s", cp->data[ret].address);printf("修改成功\n");
}//销毁通讯录
void DestoryContact(Contact* cp)
{free(cp->data);cp->data = NULL;cp->size = 0;cp->capacity = 0;
}

主函数部分(test.cpp

#define _CRT_SECURE_NO_WARNINGS 1
#include "Contact.h"//枚举,增加程序的可读性
enum options
{EXIT,ADD,DEL,SEACH,MODIFY,SHOW
};int main()
{int input = 0;//创建通讯录Contact con;//初始化通讯录InitContact(&con);do{menu();printf("请输入你的选择: ");scanf("%d", &input);switch (input){//增加联系人信息case ADD:AddContact(&con);break;//删除联系人信息case DEL:DelContact(&con);break;//查找某个联系人的信息case SEACH:SeachPeople(&con);break;//修改某个联系人的信息case MODIFY:ModifyContact(&con);break;//展示通讯录内的每一个联系人的信息case SHOW:ShowContact(&con);break;//退出通讯录管理系统case EXIT:DestoryContact(&con);printf("通讯录已退出\n");break;//预防非法输入default:printf("输入错误,请重新输入\n");break;}} while (input);return 0;
}

本次分享就到此为止了,感谢您的支持,如有错误,欢迎积极指正

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

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

相关文章

山海鲸汽车需求调研系统:智慧决策的关键一步

随着社会的发展和科技的进步&#xff0c;汽车行业也迎来了新的挑战和机遇。如何更好地满足用户需求、提高产品竞争力成为了汽车制造商们关注的焦点。在这个背景下&#xff0c;山海鲸汽车需求调研互动系统应运而生&#xff0c;为汽车行业赋予了智慧决策的力量。 智慧决策的核心&…

Idea本地跑flink任务时,总是重复消费kafka的数据(kafka->mysql)

1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 1 Idea中执行任务时&#xff0c;没法看到JobManager的错误&#xff0c;以至于我以为是什么特殊的原因导致任务总是反复消费。在close方法中&#xff0c;增加日志&#xff0c;发现jdbc连接被关闭了。 重新…

redis 主从复制

配从不配主 主要开启密码 在redis.conf requirepass xxxx 在从机配置 查看主从关系 在主机或从机执行 info replication 注意点 从机可以读&#xff0c;但没有写的权限主从机同时启动&#xff0c; 主机写一个命令&#xff0c;从机复制一个命令 主机开启动&#xff0c;写入命…

【智慧燃气】智慧燃气解决方案总体概述--终端层、网络层

关键词&#xff1a;智慧燃气、智慧燃气系统、智慧燃气平台、智慧燃气解决方案、智慧燃气应用、智能燃气 智慧燃气解决方案是基于物联网、大数据、云计算、移动互联网等先进技术&#xff0c;结合燃气行业特征&#xff0c;通过智能设备全面感知企业生产、环境、状态等信息的全方…

linux下Qt的pro文件

生成生成文件后缀名的说明。这只是泛泛而谈&#xff0c;实际发现跟编译器有关。比如在windows系统上用MinGW&#xff0c;可能静态库还是a后缀。 文件静态库动态库目标文件LINUXasooWINDOWSlibdllobj 在.pro文件中&#xff0c;INCLUDEPATH用于引入外部库的头文件&#xff0c;L…

嵌入式面试:大疆 2023 春招

文章目录 一、笔试SPI的四种模式&#xff1a;模式0 (CPOL0; CPHA0)模式1 (CPOL0; CPHA1)模式2 (CPOL1; CPHA0)模式3 (CPOL1; CPHA1) 一、笔试 SPI的四种模式&#xff1a; CPOL (Clock POlarity)&#xff1a;决定时钟极性 CPHA (Clock PHAse)&#xff1a;决定时钟相位 modeCP…

Ubuntu 22.04 clang++ /usr/bin/ld: 找不到 -lstdc++: 没有那个文件或目录

/usr/bin/ld: 找不到 -lstdc: 没有那个文件或目录 clang-16: error: linker command failed with exit code 1 (use -v to see invocation) 解决 sudo apt install libstdc-12-dev

Linux ❀ 磁盘IO较大故障告警排查确认方法

文章目录 1、iotop2、iostat3、磁盘压力测试 问题描述&#xff1a;在日常运维工作中&#xff0c;经常会遇到Linux服务器出现Disk磁盘I/O&#xff08; I/O 英文全称是 Input/Output&#xff0c;中文译为 输入与输出&#xff0c;通常指存储器与其他设备之间的数据交换操作&#x…

Rn使用FlatList导航栏自动回到中间

import { useState, useRef } from react import { FlatList, View, Text, StyleSheet, TouchableOpacity } from react-nativeconst Center () > {const tabs ["语文", "数学", "英语", "政治", "历史", "地理&q…

工学云打卡签到自动实现关于异地时定位的问题解决|蘑菇钉

工学云打卡助手&#xff0c;能解决你在异地时每天不间断签到的问题&#xff0c;仔细看图哦 1.自动签到 2.自定义打卡地区 3.生成日周月报与总结自动发表 4.支持随机通用内容 5.支持打卡结果推送 你是否曾经因为缺乏自律而无法坚持学习目标&#xff1f;是否曾经因为无法衡量…

分布式文件服务器——初识MinIO

开篇 MinIO ——开源优秀的分布式对象存储系统。 适用于AI的 高性能分布式云存储 MinIO 提供高性能、与S3 兼容的对象存储系统&#xff0c;让你自己能够构建自己的私有云储存服务。 MinIO原生支持 Kubernetes&#xff0c;它可用于每个独立的公共云、每个 Kubernetes 发行版、私…

C++: 继承

学习目标 1.继承的概念及定义 2.基类和派生类对象赋值转换(切片) 3.继承中的作用域(隐藏/重定义) 4.派生类的默认成员函数 5.继承与友元 6.继承与静态成员 7.菱形继承与菱形虚拟继承 8.总结 1.继承的概念及定义 1.1概念 继承: 它允许你创建一个新的类&#xff08;称为子类或派…

数据结构之堆的实现

首先我们要想堆应该用什么实现&#xff0c;我们这里可以使用数组&#xff0c;因为每一层的数是有限个的&#xff0c;所以我们可以很容易将数的每一个位置对应到数组中去。 那我们就可以仿照顺序表写出我们的结构。 然后我们要考虑我们要实现哪些函数。 最基本的功能初始化和销…

rabbitmq查看节点信息命令失败

不影响访问rabbitmq&#xff0c;但是无法使用 命令查看节点信息 等 查看节点信息命令&#xff1a;rabbitmq-diagnostics status --node rabbitJHComputer Error: unable to perform an operation on node ‘rabbitJHComputer‘. Please see diagnostics informatio rabbitmq-…

ssm+vue的台球厅管理系统(有报告)。Javaee项目,ssm vue前后端分离项目。

演示视频&#xff1a; ssmvue的台球厅管理系统(有报告)。Javaee项目&#xff0c;ssm vue前后端分离项目。 项目介绍&#xff1a; 采用M&#xff08;model&#xff09;V&#xff08;view&#xff09;C&#xff08;controller&#xff09;三层体系结构&#xff0c;通过Spring S…

服务器数据恢复-V7000存储2块磁盘故障脱机的数据恢复案例

服务器数据恢复环境&#xff1a; P740AIXSybaseV7000存储&#xff0c;存储阵列柜上共12块SAS机械硬盘&#xff08;其中一块为热备盘&#xff09;。 服务器故障&#xff1a; 存储阵列柜中有磁盘出现故障&#xff0c;工作人员发现后更换磁盘&#xff0c;新更换的磁盘数据同步到40…

C++练习题。。。

499 123 456 789 499是三位数 <499的素数有: 2 3 5 7 11 13 17 19 23 29 31 37 41 43 47 53 59 61 67 71 73 79 83 89 97 101 103 107 109 113 127 131 137 139 149 151 157 163 167 173 179 181 191 193 197 199 211 223 227 229 233 239 241 251 257 263 269 271 277 281 …

遥感云大数据在灾害、水体与湿地领域典型案 例实践及 GPT 模型应用

近年来遥感技术得到了突飞猛进的发展&#xff0c;航天、航空、临近空间等多遥感平台不断增加&#xff0c;数据的空间、时间、光谱分辨率不断提高&#xff0c;数据量猛增&#xff0c;遥感数据已经越来越具有大数据特征。遥感大数据的出现为相关研究提供了前所未有的机遇&#xf…

Vue3 + Ts实现NPM插件 - 定制loading

目录 你的 Loading&#x1f916; 安装&#x1f6f9; 简介苍白请 您移步文档&#xff1a;✈️ 使用方法&#x1f6e0;️ 配置 loading 类型&#x1f3b2; 定制 loading 色彩 &#x1f4a1; 注意事项 前期回顾 你的 Loading 开箱即可用的 loading&#xff0c; 说明&#xff1a;vu…

基于多线程的Reactor模式的 回声服务器 EchoServer

记录下 一个线程专门用来接受accept获取客户端的fd 获取fd之后 从剩余的执行线程中 找到一个连接客户端数量最少的线程 然后将客户端的fd加入到这个线程中并通过EPOLL监听这个fd 线程之间通过eventfd来通信 将客户端的fd传到 对应的线程中 参考了MediaServer 引入…