【C语言】实现通讯录管理系统

大家好,我是苏貝,本篇博客带大家实现通讯录,如果你觉得我写的还不错的话,可以给我一个赞👍吗,感谢❤️
在这里插入图片描述


目录

  • 一. 前言
  • 二. 通讯录的实现
    • 2.1 写出基本框架
    • 2.2 制作menu菜单
    • 2.3 创建联系人和通讯录结构体
    • 2.4 定义通讯录并初始化
    • 2.5 使用枚举常量优化代码
    • 2.6 增加联系人信息
    • 2.7 删除联系人信息
    • 2.8 查找联系人信息
    • 2.9 修改联系人信息
    • 2.10 打印所有联系人信息
    • 2.11 以名字排序所有联系人
  • 三. 模块化代码实现
    • 3.1 test.c
    • 3.2 contact.h
    • 3.3 contact.c
    • 3.4 结果演示
  • 四. 存在的问题

一. 前言

本文将使用C语言来实现通讯录管理系统,该通讯录包括若干联系人,和每个联系人的姓名、年龄、性别、电话、地址。此通讯录的功能包括:增加联系人信息,删除联系人信息,查找联系人信息,修改联系人信息,显示所有联系人信息,根据名字顺序给所有联系人信息排序

二. 通讯录的实现

首先,我们建立1个contact.h文件,即一个头文件,用来声明。建立1个test.c文件,用来测试代码。再建立1个contact.c文件,用来写实现上述6个功能的代码

2.1 写出基本框架

下面的代码就是通讯录的基本框架,很多程序的基本框架和下面代码相似,用do-while循环来决定实现什么功能。变量input作为循环的判断条件,如果选择0,则退出循环,选择1~6之间的数则执行对应的功能,选择其它数则提示输入错误,再进入循环

int main()
{int input = 0;do{menu();printf("请选择:");scanf("%d", input);switch (input){case 1:break;case 2:break;case 3:break;case 4:break;case 5:break;case 6:break;case 0:printf("退出程序\n");break;default:printf("选择错误,请重新输入\n");break;}} while (input);return 0;
}

2.2 制作menu菜单

基本框架中有menu菜单,那让我们来了解一下如何写menu菜单,即一个名为menu的函数。菜单界面就像餐厅里服务员递给你的菜单,根据菜单里的内容进行选择。选择1~6,实现特定功能;选择0,退出程序

void menu()
{printf("-----------------------------------\n");printf("***********************************\n");printf("******  1.Add       2.Del    ******\n");printf("******  3.Select    4.Modify ******\n");printf("******  5.Show      6.Sort   ******\n");printf("******  0.exit               ******\n");printf("***********************************\n");printf("-----------------------------------\n");
}

2.3 创建联系人和通讯录结构体

因为联系人信息包括姓名、年龄、性别、电话、地址,而且它们的类型和所占大小不同,所以我们想到构建一个联系人结构体,为了书写分别,我们将struct PeoInfo重命名为PeoInfo。

typedef struct PeoInfo
{char name[20];int age;char gender[5];char tele[12];char addr[30];
}PeoInfo;

现在我们将上面字符数组都确定了最大元素个数,那如果有一天我们发现有一个人的地址不止30个字节怎么办呢?我们接下来的代码里面也可能有addr[30],所以以后修改起来不太方便。因此,不妨我们用define定义一些标识符常量,之后敲代码时需要用到字符大小就写标识符常量而非一个数字,这样可以方便以后修改数组的大小

#define NAME_MAX 20
#define GENDER_MAX 5
#define TELE_MAX 30
#define ADDR_MAX 30typedef struct PeoInfo
{char name[NAME_MAX];int age;char gender[GENDER_MAX];char tele[TELE_MAX];char addr[ADDR_MAX];
}PeoInfo;

如果我们想存储若干个联系人的信息,那我们就要定义一个数组,数组有若干个元素,元素的类型是PeoInfo,很自然的,我们用define定义的标识符常量来表示数组的最大元素个数。通讯录除了要有所有联系人信息外,也应该可以告诉我们有多少个联系人,所以通讯录里要有2个东西,一个是若干联系人信息,一个是联系人的个数,它们的类型不同,所以用结构体表示通讯录

#define MAX 1000
typedef struct Contact
{PeoInfo data[MAX];int sz;
}Contact;

2.3 的都是声明语句,所以都放在contact.h文件中


2.4 定义通讯录并初始化

通讯录的定义简简单单,不用多说

	Contact con;

我们来想想如何将它初始化。(如果不初始化,那么con里面的内容都是随机值)
在test.c文件中调用初始化函数,在contact.h中声明初始化函数,在contact.c文件中定义函数。结构体传参时最好是传址而非传值,且初始化需要改变结构体的内容,所以只能传址

	InitContact(&con);

初始化就是将变量con的元素sz的值赋值为0,并且data数组里面的元素也都是0。sz不用多说,将data数组里面的元素全变为0,只需应用我们之前介绍过的memset函数了解memset函数

void InitContact(Contact* pc)
{assert(pc);//断言pc->sz = 0;memset(pc->data, 0, sizeof(pc->data));
}

2.5 使用枚举常量优化代码

如果不让你看菜单,你能知道上面基本框架的switch语句中的case 1~6分别是什么意思吗?如果你和我一样不太记得的话,那我们可以使用枚举常量来帮助我们识别。枚举常量如果我们自己不去主动赋值的话,它们的值会从0开始,依次+1,所以EXIT的值为0,ADD的值为1……

enum Option
{EXIT,//0ADD,DEL,SELECT,MODIFY,SHOW,SORT
};

所以switch语句中的case语句后面的值可以用带提示的枚举常量代替

switch (input){case ADD:break;case DEL:break;case SELECT:break;case MODIFY:break;case SHOW:break;case SORT:break;case EXIT:printf("退出程序\n");break;default:printf("选择错误,请重新选择\n");break;}

2.6 增加联系人信息

在test.c文件中调用函数

case ADD:AddContact(&con);break;

在contact.c文件中定义函数。先判断通讯录是否满了,若没满则输入联系人信息,最好一定要记得sz++。注意:输入年龄的scanf语句中,记得要&

void AddContact(Contact* pc)
{assert(pc);if (pc->sz == MAX){printf("通讯录已满,增加失败\n");return;}printf("请输入名字:");scanf("%s",pc->data[pc->sz].name );printf("请输入年龄:");scanf("%d", &(pc->data[pc->sz].age));printf("请输入性别:");scanf("%s", pc->data[pc->sz].gender);printf("请输入电话:");scanf("%s", pc->data[pc->sz].tele);printf("请输入地址:");scanf("%s", pc->data[pc->sz].addr);pc->sz++;printf("增加成功\n");
}

2.7 删除联系人信息

在test.c文件中调用函数

case DEL:DelContact(&con);break;

在contact.c文件中定义函数。先判断通讯录是否为空,不为空时,输入要删除的人的姓名,在通过for循环搜索是否有该姓名的联系人,因为除了删除联系人信息功能需要搜索以外,查找和修改联系人信息也需要通过for循环搜索是否有该姓名的联系人,所以不妨将其封装成一个函数。若没有找到这个姓名的联系人,返回-1,否则返回该姓名的下标。
找到该姓名的联系人以后做删除操作,删除操作只需要让下标后面的元素依次向前进一位,覆盖掉下标的位置即可。所以循环的结束条件是i < pc->sz - 1,不要搞错。覆盖时只需要pc->data[i] = pc->data[i + 1];因为如果结构体的类型完全相同,那就可以整体赋值。最后sz–。可能有人注意到了,这个删除操作不能删除最后一个元素,是的,但没有关系,因为我们的sz–了,所以之后的遍历中不会再遍历原本的最后一个元素

int FindOfName(char* name, Contact* pc)
{assert(name && pc);int i = 0;for (i = 0; i < pc->sz; i++){if (0 == strcmp(name, (pc->data[i].name)))return i;}return -1;
}void DelContact(Contact* pc)
{assert(pc);char name[NAME_MAX] = { 0 };if (pc->sz == 0){printf("通讯录为空,删除失败\n");return;}printf("请输入要删除的人的名字:");scanf("%s", name);int ret = FindOfName(name, pc);if (ret == -1){printf("找不到要删除的人\n");return;}int i = 0;for (i = ret; i < pc->sz - 1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("删除成功\n");
}

2.8 查找联系人信息

查找联系人信息同样需要搜索函数FindOfName(name, pc);,如果要查找的人存在,那么会返回它的下标,再通过下标对这个人信息进行打印。为了直观的看出各个信息是属于哪一类的,我们可以在打印前打印提示信息,注意:年龄是%-5s,后面打印信息时是%-5d

void SelectContact(Contact* pc)
{char name[NAME_MAX] = { 0 };printf("请输入要查找的人的名字:");scanf("%s", name);int ret = FindOfName(name, pc);if (ret == -1){printf("找不到要查找的人\n");return;}printf("%-20s%-5s%-5s%-30s%-30s\n", "姓名", "年龄", "性别", "电话", "地址");printf("%-20s%-5d%-5s%-30s%-30s\n", pc->data[ret].name, pc->data[ret].age,pc->data[ret].gender, pc->data[ret].tele, pc->data[ret].addr);
}

2.9 修改联系人信息

修改联系人信息同样需要搜索函数FindOfName(name, pc);,如果要想修改的人存在,那么会返回它的下标,再通过下标对这个人的全部信息重新输入

void ModifyContact(Contact* pc)
{assert(pc);char name[NAME_MAX] = { 0 };printf("请输入要修改的人的名字:");scanf("%s", name);int ret = FindOfName(name, pc);if (ret == -1){printf("找不到要修改的人\n");return;}printf("请输入名字:");scanf("%s", pc->data[ret].name);printf("请输入年龄:");scanf("%d", &(pc->data[ret].age));printf("请输入性别:");scanf("%s", pc->data[ret].gender);printf("请输入电话:");scanf("%s", pc->data[ret].tele);printf("请输入地址:");scanf("%s", pc->data[ret].addr);
}

2.10 打印所有联系人信息

打印所有联系人信息,无非就是遍历并打印所有元素,同样加上提示信息

void ShowContact(Contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空,打印无意义\n");return;}int i = 0;printf("%-20s%-5s%-5s%-30s%-30s\n", "姓名", "年龄", "性别", "电话", "地址");for (i = 0; i < pc->sz; i++){printf("%-20s%-5d%-5s%-30s%-30s\n", pc->data[i].name, pc->data[i].age,pc->data[i].gender, pc->data[i].tele, pc->data[i].addr);}
}

2.11 以名字排序所有联系人

使用qsort函数可快速排序,了解qsort函数

int cmp(const void* a, const void* b)
{return strcmp((char*)a, (char*)b);
}void SortContact(Contact* pc)
{assert(pc);qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp);printf("排序成功\n");
}

三. 模块化代码实现

3.1 test.c

#include"contact.h"void menu()
{printf("-----------------------------------\n");printf("***********************************\n");printf("******  1.Add       2.Del    ******\n");printf("******  3.Select    4.Modify ******\n");printf("******  5.Show      6.Init   ******\n");printf("******  7.sort      0.exit   ******\n");printf("***********************************\n");printf("-----------------------------------\n");
}enum Option
{EXIT,ADD,DEL,SELECT,MODIFY,SHOW,INIT,SORT
};int main()
{Contact con;//初始化通讯录InitContact(&con);int input = 0;do{ menu();printf("请选择:");scanf("%d", &input);switch (input){case ADD:AddContact(&con);break;case DEL:DelContact(&con);break;case SELECT:SelectContact(&con);break;case MODIFY:ModifyContact(&con);break;case SHOW:ShowContact(&con);break;case INIT:InitContact(&con);break;case SORT:SortContact(&con);break;case EXIT:printf("退出程序\n");break;default:printf("选择错误,请重新选择\n");break;}} while (input);return 0;
}

3.2 contact.h

#pragma once#include<stdio.h>
#include<assert.h>
#include<string.h>#define NAME_MAX 20
#define GENDER_MAX 5
#define TELE_MAX 30
#define ADDR_MAX 30
#define MAX 1000typedef struct PeoInfo
{char name[NAME_MAX];int age;char gender[GENDER_MAX];char tele[TELE_MAX];char addr[ADDR_MAX];
}PeoInfo;typedef struct Contact
{PeoInfo data[MAX];int sz;
}Contact;//初始化通讯录
void InitContact(Contact* pc);//增加联系人
void AddContact(Contact* pc);//删除联系人
void DelContact(Contact* pc);//打印通讯录
void ShowContact(Contact* pc);//查找联系人
void SelectContact(Contact* pc);//修改联系人信息
void ModifyContact(Contact* pc);//以名字排序所有联系人
void SortContact(Contact* pc);

3.3 contact.c

#include"contact.h"//初始化通讯录
void InitContact(Contact* pc)
{assert(pc);pc->sz = 0;memset(pc->data, 0, sizeof(pc->data));
}//增加联系人
void AddContact(Contact* pc)
{assert(pc);if (pc->sz == MAX){printf("通讯录已满,增加失败\n");return;}printf("请输入名字:");scanf("%s",pc->data[pc->sz].name );printf("请输入年龄:");scanf("%d", &(pc->data[pc->sz].age));printf("请输入性别:");scanf("%s", pc->data[pc->sz].gender);printf("请输入电话:");scanf("%s", pc->data[pc->sz].tele);printf("请输入地址:");scanf("%s", pc->data[pc->sz].addr);pc->sz++;printf("增加成功\n");
}int FindOfName(char* name, Contact* pc)
{assert(name && pc);int i = 0;for (i = 0; i < pc->sz; i++){if (0 == strcmp(name, (pc->data[i].name)))return i;}return -1;
}//删除联系人
void DelContact(Contact* pc)
{assert(pc);char name[NAME_MAX] = { 0 };if (pc->sz == 0){printf("通讯录为空,删除失败\n");return;}printf("请输入要删除的人的名字:");scanf("%s", name);int ret = FindOfName(name, pc);if (ret == -1){printf("找不到要删除的人\n");return;}int i = 0;for (i = ret; i < pc->sz - 1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("删除成功\n");
}//打印通讯录
void ShowContact(Contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空,打印无意义\n");return;}int i = 0;printf("%-20s%-5s%-5s%-30s%-30s\n", "姓名", "年龄", "性别", "电话", "地址");for (i = 0; i < pc->sz; i++){printf("%-20s%-5d%-5s%-30s%-30s\n", pc->data[i].name, pc->data[i].age,pc->data[i].gender, pc->data[i].tele, pc->data[i].addr);}
}//查找联系人
void SelectContact(Contact* pc)
{char name[NAME_MAX] = { 0 };printf("请输入要查找的人的名字:");scanf("%s", name);int ret = FindOfName(name, pc);if (ret == -1){printf("找不到要查找的人\n");return;}printf("%-20s%-5s%-5s%-30s%-30s\n", "姓名", "年龄", "性别", "电话", "地址");printf("%-20s%-5d%-5s%-30s%-30s\n", pc->data[ret].name, pc->data[ret].age,pc->data[ret].gender, pc->data[ret].tele, pc->data[ret].addr);
}//修改联系人信息
void ModifyContact(Contact* pc)
{assert(pc);char name[NAME_MAX] = { 0 };printf("请输入要修改的人的名字:");scanf("%s", name);int ret = FindOfName(name, pc);if (ret == -1){printf("找不到要修改的人\n");return;}printf("请输入名字:");scanf("%s", pc->data[ret].name);printf("请输入年龄:");scanf("%d", &(pc->data[ret].age));printf("请输入性别:");scanf("%s", pc->data[ret].gender);printf("请输入电话:");scanf("%s", pc->data[ret].tele);printf("请输入地址:");scanf("%s", pc->data[ret].addr);
}int cmp(const void* a, const void* b)
{return strcmp((char*)a, (char*)b);
}//以名字排序所有联系人
void SortContact(Contact* pc)
{assert(pc);qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp);printf("排序成功\n");
}

3.4 结果演示

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述


四. 存在的问题

前面实现的通讯录有什么问题呢?
1.录入的信息,等程序结束后,还存在吗?不存在了,因为数据是存放内存中的,只要程序退出,掉电,都会
丢失怎么解决呢?用文件存储,数据库…
2.通讯录的大小是固定的100个元素,存放100个人的信息如果信息太多,空间小了;如果信息太少,空间大了。怎么解决?用动态内存管理!
所以期待博主的下一篇优化的通讯录管理系统吧


好了,那么本篇博客就到此结束了,如果你觉得本篇博客对你有些帮助,可以给个大大的赞👍吗,感谢看到这里,我们下篇博客见❤️

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

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

相关文章

LSM Tree 深度解析

我们将深入探讨日志结构合并树&#xff0c;也称为LSM Tree&#xff1a;这是许多高度可扩展的NoSQL分布式键值型数据库的基础数据结构&#xff0c;例如Amazon的DynamoDB、Cassandra和ScyllaDB。这些数据库的设计被认为支持比传统关系数据库更高的写入速率。我们将看到LSM Tree如…

驱动开发4 使用字符设备驱动的分步实现编写LED驱动(LED亮灯)

一、思维导图 二、通过字符设备驱动的分步实现编写LED驱动&#xff0c;另外实现特备文件和设备的绑定 应用程序 test.c #include<stdlib.h> #include<stdio.h> #include <sys/types.h> #include <sys/stat.h> #include <sys/ioctl.h> #include…

【python】接入MySQL实际操作案例

Python程序接入MySQL数据库 文章目录 Python程序接入MySQL数据库建库建表接入MySQL代码实操插入数据删除数据更新数据查询数据 案例讲解 在 Python3 中&#xff0c;我们可以使用 mysqlclient或者 pymysql三方库来接入 MySQL 数据库并实现数据持久化操作。二者的用法完全相同&…

Jenkins部署失败:JDK ‘jdk1.8.0_381‘ not supported to run Maven projects

Jenkins部署报错&#xff1a;JDK ‘jdk1.8.0_381’ not supported to run Maven projects提示使用的jdk有问题&#xff0c;启动的jdk版本不能满足项目启动。 登录Jenkins管理页面&#xff0c;系统管理——全局工具配置——JDK安装配置满足条件的JDK版本&#xff0c;保存配置&…

Parallels Client for Mac:改变您远程控制体验的革命性软件

在当今数字化的世界中&#xff0c;远程控制软件已经成为我们日常生活和工作中不可或缺的一部分。在众多远程控制软件中&#xff0c;Parallels Client for Mac以其独特的功能和出色的性能脱颖而出&#xff0c;让远程控制变得更加简单、高效和灵活。 Parallels Client for Mac是…

实现el-table打印功能,样式对齐,去除滚动条

实现el-table打印功能,样式对齐&#xff0c;去除滚动条 // 整个页面打印 function printTable(id) {// let domId #js_index// if (id) {// domId #${ id };// }// let wpt document.querySelector(domId);// let newContent wpt.innerHTML;// let oldContent document.…

08数据结构——排序

8.2 插入排序 8.2.1 直接插入排序 直接插入排序&#xff08;用哨兵&#xff09;代码如下&#xff1a; void InsertSort(ElemType A[],int n){int i,j;for(i2;i<n;i) //依次将A[2]~A[n]插入前面已排序序列if(A[i]<A[i-1]){ //若A[i]关键码小于其前驱…

成都瀚网科技有限公司抖音小店:创新营销引领电商潮流

在当今数字化时代&#xff0c;抖音作为一款备受欢迎的短视频平台&#xff0c;不仅吸引了大量用户的关注&#xff0c;还为众多电商企业提供了新的销售渠道。成都瀚网科技有限公司抖音小店便是其中之一&#xff0c;凭借其独特的营销策略和优质的产品&#xff0c;成为了抖音电商领…

PyCharm改变代码背景图片的使用教程

一个好的集成环境是学习和使用一门编程语言的重中之重&#xff0c;这次我给大家分享如何改变PyCharm软件的代码背景图片。 说明&#xff1a;本教程使用的是汉化版PyCharm软件。 打开PyCharm软件。 点击软件最上方导航栏的文件&#xff0c;然后找到设置。 打开设置然后点击外观…

Baumer工业相机堡盟工业相机如何使用BGAPISDK生成视频(C++)

Baumer工业相机 Baumer工业相机堡盟相机是一种高性能、高质量的工业相机&#xff0c;可用于各种应用场景&#xff0c;如物体检测、计数和识别、运动分析和图像处理 Baumer的万兆网相机拥有出色的图像处理性能&#xff0c;可以实时传输高分辨率图像。此外&#xff0c;该相机还…

总结使用React做过的一些优化

一、修改css模拟v-show {!flag && <MyComponent style{{display: none}} />} {flag && <MyComponent />}<MyComponent style{{ display: flag ? block : none }} />二、循环使用key const todosList todos.map(item > {<li key{it…

Web APIs——焦点事件以及小米搜索框

一、事件类型 二、焦点事件 <body><input type"text"><script>const input document.querySelector(input)input.addEventListener(focus,function(){console.log(有焦点触发)})input.addEventListener(blur,function(){console.log(失去焦点触…

JS数组扁平化多维数组变为一维数组的三种方法

在实践中&#xff0c;我们有一个多维数组&#xff0c;我想把它变为一维数组&#xff0c;我们该怎么办呢&#xff1f; 让我为大家介绍一下吧&#xff01; 1.flat方法 我们使用flat可以实现降维 // 声明一个数组const arr [1,2,3,[4,5,6]]// 我们使用flat方法console.log(arr.…

hbase和aerospike基础概念及所对应的python包API使用

Hbase Hbase shell常用操作 1.创建表 create table name,column familytable name&#xff1a;表名 column family:列族名 2.查看所有表名称 list3.插入操作 put table name,row1,column family:column name,valuerow1:行键(即Row Key) column family:column name&#xf…

如何使用 JMeter 进行 HTTPS 请求测试?

本文将介绍如何使用 JMeter 测试 HTTPS 请求&#xff0c;并提供相关的技巧和注意事项。 在进行性能测试时&#xff0c;很多网站都采用了 HTTPS 协议。当我们测试 HTTPS 请求&#xff0c;如果服务端开启了双向认证&#xff0c;则需要客户端发送请求时带上证书。本文介绍如何在 …

【Axure视频教程】曲线图

今天教大家在Axure制作可视化曲线图的原型模板&#xff0c;鼠标移入曲线图后&#xff0c;会显示弹窗并回显对应折点的具体数据&#xff0c;该模板是用Axure原生元件制作的&#xff0c;所以使用方便&#xff0c;可以任意修改对应样式或者交互效果。该原型模板的具体效果可以参考…

Oracle通过局域网进行连接访问的设置

&#x1f341; 作者&#xff1a;知识浅谈&#xff0c;CSDN签约讲师&#xff0c;CSDN博客专家&#xff0c;华为云云享专家&#xff0c;阿里云专家博主 &#x1f4cc; 擅长领域&#xff1a;全栈工程师、爬虫、ACM算法 &#x1f492; 公众号&#xff1a;知识浅谈 &#x1f525;网站…

Pytorch--3.使用CNN和LSTM对数据进行预测

这个系列前面的文章我们学会了使用全连接层来做简单的回归任务&#xff0c;但是在现实情况里&#xff0c;我们不仅需要做回归&#xff0c;可能还需要做预测工作。同时&#xff0c;我们的数据可能在时空上有着联系&#xff0c;但是简单的全连接层并不能满足我们的需求&#xff0…

docker运行elastic和kibana,并使用密码连接

1. elasticsearch 运行容器 docker run -d --name elasticsearch -p 9200:9200 -p 9300:9300 -e "discovery.typesingle-node" elasticsearch:7.7.0 进入容器 docker exec -it elasticsearch bash修改配置开启密码校验 vi ./conf/elasticsearch.yml添加以下内容 …

PX4天大bug,上电反复重启,连不上QGC!

一、Debug与Bug 由于自己写的代码CPU占用率过高&#xff0c;解锁报错 CPU load too high!无法解锁。 于是把 COM_CPU_MAX 从默认的 90% 变为 99%&#xff08;千万别这样搞&#xff0c;这是bug&#xff0c;除非想玩&#xff01;&#xff09;。 然后重启&#xff0c;飞机就反…