【C语言】通讯录系统实现
目录
1、通讯录系统介绍
2、代码分装
3、代码实现步骤
3.1制作菜单函数以及游戏运行逻辑流程
3.2、封装人的信息PeoInfo以及通讯录Contact结构体类型
3.3、初始化通讯录InitContact函数
3.4、增加联系人AddContact函数
3.5、显示所有联系人ShowContact函数
3.6、删除联系人DelContact函数以及判断是否存在FindByName函数
3.7、查找指定联系人SearchContact函数
3.8、修改指定联系人ModifyContact函数
3.9、以年龄排序联系人SortContact函数
4、使用动态规划优化通讯录
1、通讯录系统介绍
实现一个通讯录:
- 可以保存100个人的信息(后续优化成动态开辟)
- 增加人的信息
- 删除指定联系人的信息
- 修改指定联系人的信息
- 查询指定联系人的信息
- 显示所有联系人的信息
- 排序通讯录的信息
其中,人的信息包括:名字、年龄、性别、电话 、地址
2、代码分装
源文件:tect.c 测试通讯录的基本功能
contact.c 相关函数的实现
头文件 contact.h相关函数和类型的声明,包含要引用的头文件和宏
3、代码实现步骤
3.1制作菜单函数以及游戏运行逻辑流程
在 test.c 中定义一个menu函数打印菜单,提示玩家进行选择,增删查改等选线;
scanf接收玩家输入并用switch判断,针对判断进行相应操作,输入错误时提示选择错误,重新选择。用到do while语句:为了能够让用户重新选择以及完成一个功能操作时再继续下一个功能操作,需要使用do while语句将它们包含起来。
使用枚举一一列举菜单选择的可能取值,便于后续使用时能够见名知意,增加代码可读性,而且枚举变量的值是默认从0开始。
【test.c】
void menu()
{printf("***********************************\n");printf("****** 1.add 2.del ******\n");printf("****** 3.search 4.modify ******\n");printf("****** 5.show 6.sort ******\n");printf("****** 0.exit ******\n");printf("***********************************\n");
}enum Option //枚举常量
{EXIT,ADD,DEL,SEARCH,MODIFY,SHOW,SORT
};int main()
{int input = 0;do{menu();printf("请输入你的选择:>");scanf("%d", &input);switch (input){case ADD:break;case DEL:break;case SEARCH:break;case MODIFY:break;case SHOW:break;case SORT:break;case EXIT:printf("退出通讯录\n");break;default:printf("选择错误,重新选择\n");break;}} while (input);return 0;
}
3.2、封装人的信息PeoInfo以及通讯录Contact结构体类型
一个人的信息包括很多,名字、年龄、性别、电话 、地址。这些成员是不同类型,所以创建一个结构体变量来保存人的信息Peoinfo。
同理通讯录也是使用一个结构体,要包括人的信息,也要知道已经存入多少个人的信息了。
【contact.h】
#define NAME_MAX 20
#define SEX_MAX 5
#define TELE_MAX 12
#define ADDR_MAX 30#define MAX 100typedef struct PeoInfo
{char name[NAME_MAX];int age;char sex[SEX_MAX];char tele[TELE_MAX];char addr[ADDR_MAX];
}PeoInfo;typedef struct Contact
{PeoInfo data[MAX];int sz; //记录的是当前通讯录中存放的人的信息个数
}Contact;
Contact这个结构体类型中包含了PeoInfo类型的数组data,数组的元素是 PeoInfo类型。
还包括sz用于记录当前保存了多少个联系人的信息。

3.3、初始化通讯录InitContact函数
【contact.h】
//初始化通讯录
void InitContact(Contact* pc);
【contact.c】
void InitContact(Contact* pc)
{assert(pc);pc->sz = 0;memset(pc->data, 0, sizeof(pc->data));
}
在contact.c中具体的实现这个函数,采用指针+箭头操作符来访问初始化con中的data成员。

将data和sz初始化为0
assert断言,对传入的指针进行判断,防止对空指针进行操作。
使用 memset函数 初始化结构体中的数据。
【test.c】
int main()
{int input = 0;Contact con; //通讯录InitContact(&con);//初始化通讯录do{menu();printf("请输入你的选择:>");scanf("%d", &input);switch (input){case ADD:break;case DEL:break;case SEARCH:break;case MODIFY:break;case SHOW:break;case SORT:break;case EXIT:printf("退出通讯录\n");break;default:printf("选择错误,重新选择\n");break;}} while (input);return 0;
}
创建了一个Contact类型的结构体变量con,调用初始化函数,把con的地址作为参数传出
在contact.c中具体的实现这个函数,采用指针+箭头操作符来访问初始化con中的data成员。
3.4、增加联系人AddContact函数
同样是函数的声明
函数的实现
函数的调用
函数的声明【contact.h】
//增加联系人
void AddContact(Contact* pc);
函数的实现【contact.c】
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].sex);printf("请输入电话:");scanf("%s", pc->data[pc->sz].tele);printf("请输入地址:");scanf("%s", pc->data[pc->sz].addr);pc->sz++;printf("增加成功\n");}
pc->data[pc->sz].name就是访问data数组中的第sz个元素的name。也就是用户要添加的是第几个联系人,他的名字,年龄,性别....
3.5、显示所有联系人ShowContact函数
【contact.h】
//显示所有联系人
void ShowContact(const Contact* pc);
【contact.c】
void ShowContact(const Contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空,无需打印\n");return;}int i = 0;printf("%-20s%-5s%-5s%-12s%-30s\n", "名字", "年龄", "性别", "电话", "地址"); //标题行for ( i = 0; i < pc->sz; i++){printf("%-20s%-5d%-5s%-12s%-30s\n",pc->data[i].name, pc->data[i].age, pc->data[i].sex, pc->data[i].tele, pc->data[i].addr);}
}
- 名为ShowContact的函数,用于打印通讯录对象的内容。函数的参数是一个指向const Contact类型的指针。
- 在函数内部,首先使用assert宏对传入的指针进行断言,确保指针非空。然后判断通讯录的大小(sz字段)是否为0,如果为0则说明通讯录为空,打印提示信息并返回。
- 接下来使用一个循环来遍历通讯录的每个联系人对象,并使用printf函数按照一定的格式打印联系人的信息。第一次循环打印标题行,后续循环每次打印一个联系人的信息。
- 打印的信息包括:名字(name)、年龄(age)、性别(sex)、电话(tele)和地址(addr)。
【test.c】
在switch语句中的case show调用函数
switch (input){case ADD:AddContact(&con);break;case DEL:break;case SEARCH:break;case MODIFY:break;case SHOW:ShowContact(&con);break;case SORT:break;case EXIT:printf("退出通讯录\n");break;default:printf("选择错误,重新选择\n");break;}

3.6、删除联系人DelContact函数以及判断是否存在FindByName函数
【contact.h】
//删除联系人
void DelContact(Contact* pc);
【contact.c】
- 因为通过名字判断此人是否存在的FindByName函数这个功能在其他操作上也需要使用到,所以最好将它封装成一个函数,减少代码冗余并且提高写代码效率。
- 当FindByName函数在通讯录中找到此人时返回此人在data数组中的下标,找不到是返回-1。返回-1就是为假
- 又因为FindByName函数只在contact.c中被使用,因此可以加上static关键字修饰。
static int FindByName(Contact* pc, char name[])
{assert(pc);int i = 0;for ( i = 0; i < pc->sz; i++){if (strcmp(pc->data[i].name, name) == 0) //两个字符串比较用strcmpreturn i;}return -1;
}void DelContact(Contact* pc)
{char name[NAME_MAX];assert(pc);if (pc->sz == 0){printf("通讯录为空,无法删除\n");return;}printf("请输入要删除人的名字:");scanf("%s", name);//查找名字为name的人int ret = FindByName(pc, name);if (ret == -1){printf("要删除的人不存在\n");return;}//删除这个人int i = 0;for (i = ret; i < pc->sz - 1; i++) //注意此处的sz - 1 {pc->data[i] = pc->data[i + 1];}pc->sz--;printf("删除成功\n");
}
- DelContact函数中创建的char类型的数组name是为了存储要删除的联系人的名字。这是因为在循环中进行联系人比较时,我们需要将要删除的联系人的名字与contacts数组中的每个联系人的名字进行比较,以找到要删除的联系人的索引。
- 如果没有name数组并且直接在循环中将contacts[i].name与要删除的联系人的名字进行比较,那么在找到要删除的联系人之后,我们将无法在后续的循环中对其进行操作。因为我们只能访问到contacts[i]和contacts[i-1],找不到要删除的联系人的索引。
- 通过将要删除的联系人的名字存储在name数组中,我们可以在找到要删除的联系人之后,继续使用它的索引进行相应操作,例如移动后面的联系人、更新sz等等。这样就可以正确地删除联系人,并保持contacts数组的完整性。
- 在代码中,DelContact函数的最后一个循环是为了从contacts数组中删除指定的联系人。循环通过将指定位置后面的联系人逐个向前移动一位来实现。因此,循环条件是i < sz-1,即i的值需要小于sz-1。
- 这是因为删除联系人时,需要将后面的联系人向前移动一个位置。而不是用后面的一个联系人来覆盖要删除的联系人。如果循环条件是i < sz,那么最后一个联系人将无法被移动,因为没有后面的联系人可以填充它的位置。比如现在是10个联系人
- 因此,循环条件为i < sz-1确保了所有联系人都能被正确地向前移动一位,保证了联系人信息的完整性。
将ret后面的所有内容向前平移一位,就可以覆盖掉要ret指向的内容,然后再对sz--,这样变相地完成了对信息的删除。等将来又有新联系人加入时,因为sz--了所以写入信息时会覆盖掉下标为ret中的内容。
3.7、查找指定联系人SearchContact函数
【contact.h】
//查找指定联系人
void SearchContact(Contact* pc);
【contact.c】
void SearchContact(Contact* pc)
{char name[NAME_MAX];assert(pc);printf("请输入要查找人的名字:");scanf("%s", name);int ret = FindByName(pc, name);if (ret == -1){printf("要查找的人不存在\n");return;}//显示查查找到的信息printf("%-20s%-5s%-5s%-12s%-30s\n", "名字", "年龄", "性别", "电话", "地址");printf("%-20s%-5d%-5s%-12s%-30s\n",pc->data[ret].name, pc->data[ret].age, pc->data[ret].sex, pc->data[ret].tele, pc->data[ret].addr);}
3.8、修改指定联系人ModifyContact函数
【contact.h】
//修改指定联系人
void ModifyContact(Contact* pc);
【contact.c】
void ModifyContact(Contact* pc)
{char name[NAME_MAX];assert(pc);printf("请输入要修改人的名字:");scanf("%s", name);int ret = FindByName(pc, name);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].sex);printf("请输入电话:");scanf("%s", pc->data[ret].tele);printf("请输入地址:");scanf("%s", pc->data[ret].addr);
}
调用FindByName函数进行判断,有此人则返回下标,没有此人则返回-1。
找到之后直接将该下标下的所有信息都重新接收并覆盖,就完成了修改操作。
3.9、以年龄排序联系人SortContact函数
【contact.h】
//排序联系人
void SortContact(Contact* pc);
【contact.c】
int cmp_age(const void* e1, const void* e2)
{return ((PeoInfo*)e1)->age - ((PeoInfo*)e2)->age;
}void SortContact(Contact* pc)
{assert(pc);if (pc->sz == 0){printf("通讯录为空,无需排序\n");return;}qsort(pc, pc->sz, sizeof(PeoInfo), cmp_age);printf("排序成功\n");
}
仍然存在的问题:
1.录入的信息,等程序结束后就不存在了,这是因为数据存放在内存中的。为了解决这个问题,需要使用到文件存储的知识。
2.通讯录的大小是固定的100个元素,只能最多存放100个人。当信息太少时,就会导致空间剩余过大浪费空间,而当信息太多时空间又太小了无法进行存入,而解决这个问题需要使用到动态内存管理的知识。下面就来优化一下通讯录。
4、使用动态规划优化通讯录
规定:
- 通讯录刚开始时可以存放3个人的信息。#define DEFAULT_SZ 3
- 当空间放满时,自动增加容量,每次增加2个信息的空间。#define DEFAULT_INC 2
定义通讯录Contact结构体:
首先是通讯录这个结构体要修改,因为原来定死了是100人,这里使用一个PeoInfo类型的指针,空间是malloc开辟的,方便我们使用realloc来调整大小
【静态】
静态版本
typedef struct Contact
{PeoInfo data[MAX];int sz; //记录的是当前通讯录中存放的人的信息个数
}Contact;
【动态】
动态版本
typedef struct Contact
{PeoInfo* data;int sz; //记录的是当前通讯录中存放的人的信息个数int capacity; //记录的时通讯录当前的最大容量
}Contact;
sz记录的是当前通讯录中存放的人的信息个数
capacity记录的时通讯录当前的最大容量当sz==capacit就要考虑增容了
初始化通讯录InitContact函数:
使用calloc对data指针进行动态开辟空间,如果开辟失败则用perror打印错误信息
【静态】
静态版本
void InitContact(Contact* pc)
{assert(pc);pc->sz = 0;memset(pc->data, 0, sizeof(pc->data));
}
【动态】
初始化sz,capacity,data(使用calloc来开辟)就要判断返回的指针是不是为NULL
void InitContact(Contact* pc)
{assert(pc);pc->sz = 0;pc->capacity = DEFAULT_SZ; //#define DEFAULT_SZ 3pc->data = (PeoInfo*)calloc(pc->capacity, sizeof(PeoInfo));if (pc->data == NULL){perror("InitContact->calloc");return;}
}
增加联系人AddContact函数:
人满了就要增加容量判断条件是当sz==capacit就要考虑增容了。使用if语句。使用realloc函数增容,记得要先创建一个指针tmp来判断返回的指针是不是为NULL不为NULL是赋值给p
【静态】
静态版本
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].sex);printf("请输入电话:");scanf("%s", pc->data[pc->sz].tele);printf("请输入地址:");scanf("%s", pc->data[pc->sz].addr);pc->sz++;printf("增加成功\n");}
【动态】
void AddContact(Contact* pc)
{assert(pc);if (pc->sz == pc->capacity){PeoInfo* tmp = (PeoInfo*)realloc(pc->data, (pc->capacity + DEFAULT_INC) * sizeof(PeoInfo));if (tmp != NULL){pc->data = tmp;pc->capacity += DEFAULT_INC; //#define DEFAULT_INC 2printf("增容成功\n");}else{perror("AddContact->realloc");return;}}printf("请输入名字:");scanf("%s", pc->data[pc->sz].name);printf("请输入年龄:");scanf("%d", &(pc->data[pc->sz].age));printf("请输入性别:");scanf("%s", pc->data[pc->sz].sex);printf("请输入电话:");scanf("%s", pc->data[pc->sz].tele);printf("请输入地址:");scanf("%s", pc->data[pc->sz].addr);pc->sz++;printf("增加成功\n");}
pc->data就是指针这个data是一个PeoInfo类型的指针
也可以把这个增容的操作单独封装成一个函数
if (pc->sz == pc->capacity){PeoInfo* tmp = (PeoInfo*)realloc(pc->data, (pc->capacity + DEFAULT_INC) * sizeof(PeoInfo));if (tmp != NULL){pc->data = tmp;pc->capacity += DEFAULT_INC; //#define DEFAULT_INC 2printf("增容成功\n");}else{perror("AddContact->realloc");return;}}
在EXIT退出通讯录时候记得对动态开辟的空间进行free操作:
【contact.h】
//销毁通讯录
void DestroyContact(Contact* pc);
【contact.c】
void DestroyContact(Contact* pc)
{free(pc->data);pc->data = NULL;pc->sz = 0;pc->capacity = 0;
}
结束
相关文章:
【C语言】通讯录系统实现
目录 1、通讯录系统介绍 2、代码分装 3、代码实现步骤 3.1制作菜单函数以及游戏运行逻辑流程 3.2、封装人的信息PeoInfo以及通讯录Contact结构体类型 3.3、初始化通讯录InitContact函数 3.4、增加联系人AddContact函数 3.5、显示所有联系人ShowContact函数 3.6、删除联系人D…...
(delphi11最新学习资料) Object Pascal 学习笔记---第12章第1节 ( 类静态方法与Windows API回调)
12.1.4 类静态方法与Windows API回调 静态类方法没有隐藏的Self参数意味着静态类方法可以作为回调函数传递给操作系统(例如,在Windows上)。实际上,您可以声明一个具有stdcall调用约定的静态类方法,并将其用作直接的…...
第一个Rust程序
在安装好Rust以后,我们就可以编写程序了。 首先,我们执行下面的命令,尽量让你的rust版本和我的版本相同,或者比我的版本大。 zhangdapengzhangdapeng:~$ cargo --version cargo 1.78.0 (54d8815d0 2024-03-26) zhangdapengzhangd…...
【LInux】<基础IO> 文件操作 | 文件描述符 | 重定向
👦个人主页:Weraphael ✍🏻作者简介:目前正在学习c和算法 ✈️专栏:Linux 🐋 希望大家多多支持,咱一起进步!😁 如果文章有啥瑕疵,希望大佬指点一二 如果文章对…...
MySQL--增、删、改、查,
数据库的概述、发展、现状、历史、分类 MySQL关系型数据库、架构(C/S) window系统安装MySQL数据库 Linux系统【选学】 数据库对象——数据库(database) show、create、drop命令 数据库对象——表(tableÿ…...
5.12学习总结
一.JAVA聊天室项目 文件发送 使用 Java Socket 实现聊天内容或文件的传输的原理如下: 服务器端启动:聊天室的服务器端在指定的端口上监听客户端的连接。它创建一个 ServerSocket 对象,并通过调用 accept() 方法等待客户端的连接请求。客户…...
ansible利用playbook 部署lamp架构
搭建参考:ansible批量运维管理-CSDN博客 定义ansible主机清单 [rootansible-server ~]# vim /etc/hosts 192.168.200.129 host01 192.168.200.130 host02 [rootansible-server ~]# vim /etc/ansible/hosts [webserver] host01 host02 在ansible端编写index.html…...
SPI通信(使用SPI读写W25Q64)
SPI通信协议 • SPI(Serial Peripheral Interface)是由Motorola公司开发的一种通用数据总线 • 四根通信线: SCLK:串行时钟线,用来提供时钟信号的。 MOSI:主机输出,从机输入 MISO:从机输出,主机输入 SS:…...
<sa8650>QCX Usecase 使用详解—拓扑图 XML 定义
<sa8650>QCX Usecase 使用详解—拓扑图 XML 定义 一 、前言二、拓扑图 XML 定义2.1 <Node, port, link>2.2 < XML prolog >2.3 < UsecaseDef >2.4 < Usecase>2.5 < Targets>2.5.1 < Target>2.5.2 < Range>2.6 < Pipeline>2.…...
使用C++11实现Golang的defer功能
本文主要用C11标准来实现Golang的defer功能。 背景 目前笔者的主力语言是Golang,其次是C,再次是JS、Delphi。在Golang工程中大量使用了defer关键字实现函数的延迟调用。如打开文件的出错处理。近来在C工程中遇到类似需求,在函数返回时进行某…...
前端之电力系统SVG图低代码
其实所有的图形都是由点,线,面组成的。点线面可以组成一个设备。下面就简单讲讲点线面是怎么画的吧 对于线,可以用path <g><path:d"M ${beginX},${beginY} L ${endX},${endY}":stroke-width"lineWidth":strok…...
括号生成[中等]
优质博文:IT-BLOG-CN 一、题目 数字n代表生成括号的对数,请你设计一个函数,用于能够生成所有可能的并且 有效的 括号组合。 示例 1: 输入:n 3 输出:["((()))","(()())","(())(…...
配置ubuntu的VNC时遇到报错_XSERVTransmkdir: Mode of /tmp/.X11-unix should be set to 1777
现在win11内嵌了ubuntu系统,我在根据打造基于 VNC 的 Ubuntu 20.04 的远程桌面 配置VNC server时,到了 vncserver :1 这一步,遇到报错: vncserver: /usr/bin/Xtigervnc did not start up, please look into /root/.vnc/xxxxx.:1.…...
openstack部署nova中出现的问题:
[rootcontroller nova]# su -s /bin/sh -c “nova-manage db sync” nova /usr/lib/python2.7/site-packages/pymysql/cursors.py:170: Warning: (1831, u’Duplicate index block_device_mapping_instance_uuid_virtual_name_device_name_idx. This is deprecated and will be…...
【OpenCV 基础知识 3】边缘检测
文章目录 cvCanny完整示例代码 cvCanny 这行代码使用OpenCV库中的 cvCanny 函数对灰度图像进行边缘检测。让我解释一下: cvCanny(gray, dst, 10, 100, 3);gray: 这是输入的灰度图像,即要进行边缘检测的图像。dst: 这是输出的边缘图像,即将结…...
拓宽知识储备量(指数级成长)
对于增强自己的知识储备,不是什么知识都往脑袋里去塞,最好的办法就是让自己的心态回到自己初心的时候,始终保值一颗学者的心,你像那些成功人士,比如格力,华为,腾讯等这样的大公司创始人哪个不是…...
x264 帧类型代价计算原理:slicetype_mb_cost 函数分析
slicetype_mb_cost 函数 函数功能 计算每个宏块 MB 的代价 cost。函数参数分析 x264_t *h:全局编码结构体x264_mb_analysis_t *a:宏块分析结构体x264_frame_t **frames:系列帧数据结构体int p0:帧序号之一,一般指向靠前帧int p1:帧序号之一,一般指向靠后帧int b:帧标志…...
战网国际服加速器哪个好用 暴雪战网免费加速器分享
战网国际服(Battle.net International或Battle.net Global)是由暴雪娱乐公司(Blizzard Entertainment)运营的面向全球玩家的多人在线游戏平台。与专注于特定地区的版本不同,国际服允许玩家不受地域限制地访问暴雪的多款…...
Java入门基础学习笔记26——break,continue
跳转关键字: break: 跳出并结束当前所在循环的执行。 continue: 用于跳出当前循环中的当次执行,直接进入循环中的下一次执行。 package cn.ensource.loop;public class BreakContinueDemo8 {public static void main(String[] a…...
HNU-算法设计与分析-作业6
第六次作业【分支限界法】 文章目录 第六次作业【分支限界法】<1> 算法实现题6-2 最小权顶点覆盖问题<2> 算法实现题6-6 n后问题<3> 算法实现题6-7 布线问题 <1> 算法实现题6-2 最小权顶点覆盖问题 ▲问题重述 问题描述: 给定一个赋权无向…...
数据库分批入库
今天在工作中,遇到一个问题,就是分批查询的时候,由于批次过大导致出现了一些问题,一下是问题描述和解决方案: 示例: // 假设已有数据列表 dataList 和 PreparedStatement pstmt int batchSize 1000; // …...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
Pinocchio 库详解及其在足式机器人上的应用
Pinocchio 库详解及其在足式机器人上的应用 Pinocchio (Pinocchio is not only a nose) 是一个开源的 C 库,专门用于快速计算机器人模型的正向运动学、逆向运动学、雅可比矩阵、动力学和动力学导数。它主要关注效率和准确性,并提供了一个通用的框架&…...
A2A JS SDK 完整教程:快速入门指南
目录 什么是 A2A JS SDK?A2A JS 安装与设置A2A JS 核心概念创建你的第一个 A2A JS 代理A2A JS 服务端开发A2A JS 客户端使用A2A JS 高级特性A2A JS 最佳实践A2A JS 故障排除 什么是 A2A JS SDK? A2A JS SDK 是一个专为 JavaScript/TypeScript 开发者设计的强大库ÿ…...
Linux nano命令的基本使用
参考资料 GNU nanoを使いこなすnano基础 目录 一. 简介二. 文件打开2.1 普通方式打开文件2.2 只读方式打开文件 三. 文件查看3.1 打开文件时,显示行号3.2 翻页查看 四. 文件编辑4.1 Ctrl K 复制 和 Ctrl U 粘贴4.2 Alt/Esc U 撤回 五. 文件保存与退出5.1 Ctrl …...
springboot 日志类切面,接口成功记录日志,失败不记录
springboot 日志类切面,接口成功记录日志,失败不记录 自定义一个注解方法 import java.lang.annotation.ElementType; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; import java.lang.annotation.Target;/***…...
[USACO23FEB] Bakery S
题目描述 Bessie 开了一家面包店! 在她的面包店里,Bessie 有一个烤箱,可以在 t C t_C tC 的时间内生产一块饼干或在 t M t_M tM 单位时间内生产一块松糕。 ( 1 ≤ t C , t M ≤ 10 9 ) (1 \le t_C,t_M \le 10^9) (1≤tC,tM≤109)。由于空间…...
Java数组Arrays操作全攻略
Arrays类的概述 Java中的Arrays类位于java.util包中,提供了一系列静态方法用于操作数组(如排序、搜索、填充、比较等)。这些方法适用于基本类型数组和对象数组。 常用成员方法及代码示例 排序(sort) 对数组进行升序…...
字符串哈希+KMP
P10468 兔子与兔子 #include<bits/stdc.h> using namespace std; typedef unsigned long long ull; const int N 1000010; ull a[N], pw[N]; int n; ull gethash(int l, int r){return a[r] - a[l - 1] * pw[r - l 1]; } signed main(){ios::sync_with_stdio(false), …...
如何做好一份技术文档?从规划到实践的完整指南
如何做好一份技术文档?从规划到实践的完整指南 🌟 嗨,我是IRpickstars! 🌌 总有一行代码,能点亮万千星辰。 🔍 在技术的宇宙中,我愿做永不停歇的探索者。 ✨ 用代码丈量世界&…...

