【C语言进阶篇】模拟实现通讯录 (内附源码)
文章目录
- 📋 前言
- 一 、 通讯录的简介
- 1.1 联系人的类型定义
- 1.2 通讯录的定义
- 1.3 通讯录要实现的功能
- 二 、 如何实现这些功能
- 2.1 test.c 的实现
- mian() 函数的实现
- menu() 菜单函数的实现
- test() 通讯录选择的实现
- 2.2 Contact.h 的声明
- 2.3 Contact.c 功能函数的定义
- 0️⃣ 初始化通讯录
- 1️⃣ 新增联系人的实现
- 2️⃣ 删除联系人的实现
- 3️⃣查询联系人的实现
- 4️⃣ 修改联系人的实现
- 5️⃣ 查看所有联系人
- 6️⃣ 排序联系人
- 三、通讯录功能的测试
- 1️⃣ 新增联系人的测试
- 2️⃣ 删除联系人的测试
- 3️⃣查询联系人的测试
- 4️⃣ 修改联系人的测试
- 5️⃣ 查看所有联系人
- 6️⃣ 排序联系人
- 四、通讯录整体工程
- test.c
- contact.c
- contact.h
- 📝全篇总结
📋 前言
🌈hello! 各位宝子们大家好啊,结构体我们都学完了,那么我们今天就来点实战把!
⛳️给大家现编一个通讯录,其实并不难只需要用到我们的结构体知识就可以,大家一起动动手吧!
📚本期文章收录在《C语言进阶篇》,大家有兴趣可以看看呐!
⛺️ 欢迎铁汁们 ✔️ 点赞 👍 收藏 ⭐留言 📝!
🔥 注:结构体的文章在这里嗷!《结构体的万字解析》
一 、 通讯录的简介
通讯录大家可以说是在熟悉不过了,那么今天就来用我们所学的C语言知识实现一下。通讯录无非就是增加联系人和删除等,增删查改这些功能。>
- 而每个
联系人
又是不同元素
的集合- 这时我们的结构体就排上用场了
1.1 联系人的类型定义
既然是联系人,那么我们相信大家一定储存的都是联系人的:
- 姓名 年龄 性别 电话 地址
- 这些基本的元素,这些知道了我们的结构体也就可以定义了
📚 代码演示:
由于数组的数字使用起来不方便更改和没有什么特殊意义,所以我们就把这些数组可以定义的宏来。
- 可以让数组大小更容易更改
- 还可以让别人一眼就知道这些数字的含义
#define MAX 1000
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30//联系人的类型定义
typedef struct PeoInfo
{char name[MAX_NAME];int age;char sex[MAX_SEX];char tele[MAX_TELE];char addr[MAX_ADDR];
}PeoInfo;
1.2 通讯录的定义
联系人的类型我们知道了那么通讯录还不要创建嘛,首先我们需要一个数组来存放每个人的信息!和一个整形来统计我们存放了多少人。
- 那么就用 PeoInfo data[100]来负责存放100个联系人的信息。
- 用 int sz,来统计存放了多少个人
- 这里由于不知道到底存放多少个字节合适所以用宏定义一个MAX定义为100也好修改
📚 代码演示:
//通讯录的定义
#define MAX 100
typedef struct Contact
{PeoInfo data[MAX];//指向了存放数据的空间int sz;//记录当前存放的有效元素的个数
}Contact;
1.3 通讯录要实现的功能
上面我们给大家说了,通讯录的一些功能大家都清楚那么该怎么实现呢?今天就来给大家用多文件的形式一起实现一下!
- 我们来先看一下大纲是怎么样的
二 、 如何实现这些功能
2.1 test.c 的实现
在这个文件里面就是我们的主文件用来负责测试和调用函数的,
main()
函数就在次文件里面包含着!
- 首先、这个文件要完成菜单的选择以及测试函数的的调用
- 二、是操作通讯的选项实现
mian() 函数的实现
main函数的功能很简单就是调用test() 函数进行测试就好了其他什么也不用干!
📚 代码演示:
int main()
{test();return 0;}
menu() 菜单函数的实现
这里做一个简易的菜单就好了,只需要让使用者知道每个选项该怎么操作就好了!
📚 代码演示:
//菜单
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");
}
test() 通讯录选择的实现
既然是通讯录的选择,那么我们一般使用的都是多分支语句
switch
所以我们这次也按照 switch 来实现每次操作的选择。
- 由于
switch
语句里面的 1 2 3 数字的含义不是很明确- 所以我们使用枚举来列举一下,这样每个选项是什么意思
- 就一目了然了
🔥 注:不会枚举的可以看看这篇文章《枚举 联合 位段》里面有详细讲解哦!一看就懂!
📚 代码演示:
//枚举选项
enum OPTION
{EXIT,ADD, DEL, SEARCH, MODIFY, SHOW, SORT,
};//测试通讯录
void test()
{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 SEARCH:SearchContact(&con);break;case MODIFY:ModifyContact(&con);break;case SHOW:ShowContact(&con);break;case SORT:SortContact(&con);break;case EXIT:printf("退出通讯录!\n");break;default:printf("输入错误请重新输入!\n");break;}} while (input);
}
2.2 Contact.h 的声明
好了主文件我们的编写完了,接下来就是对我们所调用的函数进行声明和定义。
- 而点h 文件刚好是用来声明函数的
- 下面我们就把需要调用的函数先声明一下后面去实现
🔥 注:由于很多头文件我们,每个文件都要调用,而Contact.h 这个文件我们也需要调用。
- 所以就把一些都要用的声明提前写到点 h 的文件里面
- 例如前面的声明,和宏这些放到点 h 的文件里面使用起来方便一些
📚 代码演示:
#pragma once
#include <string.h>
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>#define MAX 1000
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30#define DEFAULT_SZ 3
#define INC_SZ 2//联系人的类型定义
typedef struct PeoInfo
{char name[MAX_NAME];int age;char sex[MAX_SEX];char tele[MAX_TELE];char addr[MAX_ADDR];
}PeoInfo;//动态通讯录
typedef struct Contact
{PeoInfo* data;//指向了存放数据的空间int sz;//记录当前存放的有效元素的个数int capacity;//通讯录当前的最大容量
}Contact;//枚举选项
enum OPTION
{EXIT,ADD, DEL, SEARCH, MODIFY, SHOW, SORT, };
//初始化通讯录函数
void InitContact(Contact* pc);
//增加联系人
void AddContact(Contact* pc);
//删除指定联系人
void DelContact(Contact* pc);
//查询联系人
void SearchContact(const Contact* pc);
//修改联系人
void ModifyContact(Contact* pc);
//显示所有联系人
void ShowContact(const Contact* pc);
//排序结构体
void SortContact(Contact* pc);
2.3 Contact.c 功能函数的定义
⛳️ 通讯录的大体框架我们都搭建起来了,接下来我们就是各种函数的实现。然后通讯录的整体工程就完成啦!
0️⃣ 初始化通讯录
这里没什么可注意的,唯一需要注意的一点就是:
- assert()断言一下确保程序的可执行性
- 和 结构体数组初始化要用
memset
函数初始化
📚 代码演示:
void InitContact(Contact* pc)
{assert(pc);memset(pc->data, 0, sizeof(pc->data));pc->sz = 0;
}
1️⃣ 新增联系人的实现
新增的大概思想就是根据我们的 sz
联系人的个数来做数组下标。然后进行访问存储数据。
- 而这里要注意的是确保一下容量会不会慢一旦满了我们就提示
- 通讯录已满,无法添加联系人。这才是最合理的思想
📚 代码演示:
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");
}
2️⃣ 删除联系人的实现
删除联系人怎么实现呢?这里是不是得先写一个查找函数先来找到我们需要删除联系人的下标然后再进行删除。
- 一 、这里要注意的意思如果联系人为空就无法删除
- 二,查找函数的实现不需要声明因为我们只需要在这一个文件下用不需要跨文件使用
- 三 、 删除完联系人后我们需要把删除的联系人的那个节点后面的联系人都给向左填充,这样我们的通讯录才又是一个有序数组!
- 还有一个重要的点是 删除完联系人,我们的有效人数 sz 也要减一
📚 代码演示:
// 查询联系人
int FindByName(const Contact* pc, char name[])
{//找到要删除的人int i = 0;for (i = 0; i < pc->sz; i++){if (strcmp(pc->data[i].name, name) == 0){return i;}}return -1;
}//删除指定联系人
void DelContact(Contact* pc)
{if (pc->sz == 0){printf("通讯录为空,无法删除!\n");return;}char name[MAX_NAME] = { 0 };assert(pc);//删除联系人printf("请输入要删除人的名字:>");scanf("%s", name);//查询联系人int del = FindByName(pc, name);if (del == -1){printf("要删除的人不存在,删除失败!\n");return;}int i = 0;for (i = del; i < pc->sz-1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("成功删除联系人!\n");
}
3️⃣查询联系人的实现
这个就比较简单了,前面我们已经简单的实现了一个查找函数用来查找下标。
- 这里需注意的是由于我们只需要查找联系人而不用修改
- 所以我们在接收只指针的时候要记得,使用
const
进行修饰- 来确保指针指向的内容不会被我们改变而引发程序错乱
📚 代码演示:
//查询联系人
void SearchContact(const Contact* pc)
{char name[MAX_NAME] = { 0 };printf("请输入要查找人的名字:>");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要查找的人不存在!\n");return;}else{printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);}}
4️⃣ 修改联系人的实现
修改联系人我们该怎么办呢?答案肯定还是使用下标的方法进行更改了!诶这里大家有没有发现我们查询函数的便捷性,所以像这种只要多次使用的功能一定要封装成函数。
- 这样使用起来就会方便很多
📚 代码演示:
//修改联系人
void ModifyContact(Contact* pc)
{char name[MAX_NAME] = { 0 };printf("请输入要修改人的名字:>");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要修改的人不存在!\n");return;}else{printf("请输入名字:->");scanf("%s", pc->data[pos].name);printf("请输入年龄:->");scanf("%d", &(pc->data[pos].age));printf("请输入性别:->");scanf("%s", pc->data[pos].sex);printf("请输入电话:->");scanf("%s", pc->data[pos].tele);printf("请输入地址:->");scanf("%s", pc->data[pos].addr);printf("联系人修改成功!\n");return;}
}
5️⃣ 查看所有联系人
这里更加简单只需要使用循环来遍历我们的数组就可以了。
- 而这里我们也是只访问并不修改,所以使用指针接收的时候
- 一定要用
const
来修饰我们的指针确保指针指向的内容不会改变
📚 代码演示:
//显示所有联系人
void ShowContact(const Contact* pc)
{int i = 0;//打印列标题printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");//打印数据for (i = 0; i < pc->sz; i++){printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n", pc->data[i].name, pc->data[i].age, pc->data[i].sex, pc->data[i].tele,pc->data[i].addr);}}
6️⃣ 排序联系人
这里就需要用到我们 库函数 qsort 函数一键排序了,十分的方便
- 不会的可以去看一下博主前面的文章,看完秒会
🔥 注:文章链接在这里。《qsort的使用详解》
📚 代码演示:
int cmp(const void* p1, const void* p2)
{return (*((PeoInfo*)p1)->name, *((PeoInfo*)p2)->name);
}//结构体比较函数
void SortContact(Contact* pc)
{qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp);printf("已按照姓名排序成功!\n");
}
三、通讯录功能的测试
所有函数都实现了那么我们就来查看一下每个功能的是否正常呢?接下来我们就来测试一下!
1️⃣ 新增联系人的测试
- 这里我们就可以看到新增联系人的函数实现成功了!
2️⃣ 删除联系人的测试
⛳️这里就可以看到本来我们是有3个联系人的,然后进删除选了就只剩俩个联系人了!
3️⃣查询联系人的测试
4️⃣ 修改联系人的测试
这里我们就把翠花的信息重新修改,为小美了,说明这个函数也实现了
5️⃣ 查看所有联系人
6️⃣ 排序联系人
四、通讯录整体工程
test.c
#define _CRT_SECURE_NO_WARNINGS 1
//实现一个通讯录
// 名字
// 年龄
// 性别
// 电话
// 地址
//
// 通讯录的功能可以存放
//增加联系人
// 删除联系人
// 修改联系人
// 查找联系人
// 显示所有联系人
// 排序功能
//
#include <stdio.h>
#include "contact.h"//菜单
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");
}//测试通讯录
void test()
{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 SEARCH:SearchContact(&con);break;case MODIFY:ModifyContact(&con);break;case SHOW:ShowContact(&con);break;case SORT:SortContact(&con);break;case EXIT:printf("退出通讯录!\n");break;default:printf("输入错误请重新输入!\n");break;}} while (input);
}
int main()
{test();return 0;}
contact.c
#define _CRT_SECURE_NO_WARNINGS 1
#include "contact.h"//初始化通讯录函数
void InitContact(Contact* pc)
{assert(pc);pc->data = (PeoInfo*)malloc(DEFAULT_SZ *sizeof(PeoInfo));if (pc->data == NULL){perror("InitContact");return;}pc->sz = 0;pc->capacity = DEFAULT_SZ;}//检查容量
int CheckCapacity(Contact* pc)
{if (pc->sz == pc->capacity){PeoInfo* ptr = (PeoInfo*)realloc(pc->data, (pc->capacity + INC_SZ) * sizeof(PeoInfo));if (ptr == NULL){perror("CheckCapacity");return 0;}else{pc->data = ptr;pc->capacity += INC_SZ;printf("增容成功!\n");return 1;}}return 1;
}
//增加联系人
void AddContact(Contact* pc)
{assert(pc);CheckCapacity(pc);if (0 == CheckCapacity(pc)){return ;}else{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 ShowContact(const Contact* pc)
{int i = 0;//打印列标题printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");//打印数据for (i = 0; i < pc->sz; i++){printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n", pc->data[i].name, pc->data[i].age, pc->data[i].sex, pc->data[i].tele,pc->data[i].addr);}}// 查询联系人
int FindByName(const Contact* pc, char name[])
{//找到要删除的人int i = 0;for (i = 0; i < pc->sz; i++){if (strcmp(pc->data[i].name, name) == 0){return i;}}return -1;
}//删除指定联系人
void DelContact(Contact* pc)
{if (pc->sz == 0){printf("通讯录为空,无法删除!\n");return;}char name[MAX_NAME] = { 0 };assert(pc);//删除联系人printf("请输入要删除人的名字:>");scanf("%s", name);//查询联系人int del = FindByName(pc, name);if (del == -1){printf("要删除的人不存在,删除失败!\n");return;}int i = 0;for (i = del; i < pc->sz-1; i++){pc->data[i] = pc->data[i + 1];}pc->sz--;printf("成功删除联系人!\n");
}//查询联系人
void SearchContact(const Contact* pc)
{char name[MAX_NAME] = { 0 };printf("请输入要查找人的名字:>");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要查找的人不存在!\n");return;}else{printf("%-20s\t%-4s\t%-5s\t%-12s\t%-30s\n", "名字", "年龄", "性别", "电话", "地址");printf("%-20s\t%-4d\t%-5s\t%-12s\t%-30s\n",pc->data[pos].name,pc->data[pos].age,pc->data[pos].sex,pc->data[pos].tele,pc->data[pos].addr);}}//修改联系人
void ModifyContact(Contact* pc)
{char name[MAX_NAME] = { 0 };printf("请输入要修改人的名字:>");scanf("%s", name);int pos = FindByName(pc, name);if (pos == -1){printf("要修改的人不存在!\n");return;}else{printf("请输入名字:->");scanf("%s", pc->data[pos].name);printf("请输入年龄:->");scanf("%d", &(pc->data[pos].age));printf("请输入性别:->");scanf("%s", pc->data[pos].sex);printf("请输入电话:->");scanf("%s", pc->data[pos].tele);printf("请输入地址:->");scanf("%s", pc->data[pos].addr);printf("联系人修改成功!\n");return;}
}//
//int cmp(const void* p1, const void* p2)
//{
// return (((Contact*)p1)->data)->age- (((Contact*)p2)->data)->age;
//}int cmp(const void* p1, const void* p2)
{return (*((PeoInfo*)p1)->name, *((PeoInfo*)p2)->name);
}//结构体比较函数
void SortContact(Contact* pc)
{qsort(pc->data, pc->sz, sizeof(pc->data[0]), cmp);printf("已按照姓名排序成功!\n");
}void DestroyContact(Contact* pc)
{free(pc->data);pc->sz = 0;pc->capacity = 0;
}
contact.h
#pragma once
#include <string.h>
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>#define MAX 1000
#define MAX_NAME 20
#define MAX_SEX 5
#define MAX_TELE 12
#define MAX_ADDR 30#define DEFAULT_SZ 3
#define INC_SZ 2//联系人的类型定义
typedef struct PeoInfo
{char name[MAX_NAME];int age;char sex[MAX_SEX];char tele[MAX_TELE];char addr[MAX_ADDR];
}PeoInfo;//动态通讯录
typedef struct Contact
{PeoInfo* data;//指向了存放数据的空间int sz;//记录当前存放的有效元素的个数int capacity;//通讯录当前的最大容量
}Contact;//枚举选项
enum OPTION
{EXIT,ADD, DEL, SEARCH, MODIFY, SHOW, SORT, };//初始化通讯录函数
void InitContact(Contact* pc);
//增加联系人
void AddContact(Contact* pc);
//删除指定联系人
void DelContact(Contact* pc);
//查询联系人
void SearchContact(const Contact* pc);
//修改联系人
void ModifyContact(Contact* pc);
//显示所有联系人
void ShowContact(const Contact* pc);
//排序结构体
void SortContact(Contact* pc);
void DestroyContact(Contact* pc);
📝全篇总结
✅ 归纳:
好了以上就是用简易通讯录的实现是不是很有趣呢!本期只用到了指针和结构体的内容!下期给大家带来动态通讯录的改造!
初始化通讯录
新增联系人的实现
删除联系人的实现
查询联系人的实现
修改联系人的实现
查看所有联系人
排序联系人
☁️ 好了把上面的知识全部掌握我相信各位铁汁们,对结构体和指针的应用更加得心应手了!
看到这里了还不给博主扣个:
⛳️ 点赞
☀️收藏
⭐️ 关注
!
💛 💙 💜 ❤️ 💚💓 💗 💕 💞 💘 💖
拜托拜托这个真的很重要!
你们的点赞就是博主更新最大的动力!
有问题可以评论或者私信呢秒回哦。
相关文章:

【C语言进阶篇】模拟实现通讯录 (内附源码)
🎬 鸽芷咕:个人主页 🔥 个人专栏:《C语言初阶篇》 《C语言进阶篇》 ⛺️生活的理想,就是为了理想的生活! 文章目录 📋 前言一 、 通讯录的简介1.1 联系人的类型定义1.2 通讯录的定义1.3 通讯录要实现的功能 二 、 如何…...

Python web实战之 Django 的模板语言详解
关键词: Python、web开发、Django、模板语言 概要 作为 Python Web 开发的框架之一,Django 提供了一套完整的 MVC 模式,其中的模板语言为开发者提供了强大的渲染和控制前端的能力。本文介绍 Django 的模板语言。 1. Django 模板语言入门 Dj…...

使用ChatGPT编写技术文档
技术文档对于任何项目都是至关重要的,因为它确保所有利益相关者都在同一层面上,并允许有效的沟通和协作。创建详细而准确的技术文档可能既耗时又具有挑战性,特别是对于那些不熟悉主题或缺乏强大写作技巧的人来说。ChatGPT 是一个强大的人工智…...
Java超级玛丽小游戏制作过程讲解 第四天 创建并完成常量类03
今天继续来完成常量类。 package com.sxt;import javax.imageio.ImageIO; import java.awt.image.BufferedImage; import java.io.File; import java.io.IOException; import java.util.ArrayList; import java.util.List;public class StaticValue {//背景public static Buff…...

webpack基础知识八:说说如何借助webpack来优化前端性能?
一、背景 随着前端的项目逐渐扩大,必然会带来的一个问题就是性能 尤其在大型复杂的项目中,前端业务可能因为一个小小的数据依赖,导致整个页面卡顿甚至奔溃 一般项目在完成后,会通过webpack进行打包,利用webpack对前…...

JAVA SE -- 第十五天
(全部来自“韩顺平教育”) 多线程 一、线程相关概念 1、程序:是为完成特定任务、用某种语言编写的一组指令的集合。 2、进程:是指运行中的程序,如QQ,就启动了一个进程,操作系统就会为该进程…...

macOS 环境变量加载探究
使用 macOS 安装环境,见到过很数种环境变量配置方法,每次也都是按照别人的代码,人家配置在哪 我就配置在哪,其实不太清楚有什么区别,决定记录下。 本机 macOS 13.3,从 macOS Catalina(10.15) 开始…...
在程序中如何判断该线程的线程id(get_id())的返回值是一个无效值
std::thread::id() 是std::thread::id的默认构造函数,它会创建一个空的std::thread::id对象。一个空的std::thread::id对象代表一个无效的线程标识符。 可以通过 std::thread::id 的成员函数 std::thread::id::operator() 来判断一个 std::thread::id 是否是一个空值…...

ffmpeg-ffplay代码架构简述
全局变量 /* Minimum SDL audio buffer size, in samples. */ // 最小音频缓冲 #define SDL_AUDIO_MIN_BUFFER_SIZE 512 /* Calculate actual buffer size keeping in mind not cause too frequent audio callbacks */ // 计算实际音频缓冲大小,并不需要太频繁…...
⛳ 多线程面试-什么是多线程上下文切换?
目录 ⛳ 多线程面试-什么是多线程上下文切换?🎁 Java中用到的线程调度算法是什么?🎨 什么是线程饥饿 ?你对线程优先级的理解是什么? ⛳ 多线程面试-什么是多线程上下文切换ÿ…...
vb+SQL车辆管理系统设计与实现
摘 要 随着信息时代的到来,信息高速公路的兴起,全球信息化进入了一个新的发展时期。人们越来越认识到计算机强大的信息模块处理功能,使之成为信息产业的基础和支柱。 我国经济的快速发展,汽车已经成为人们不可缺少的交通工具。对于拥有大量车辆的机关企事业来说,车辆的…...
java的枚举类
枚举类的概念和使用 1.枚举类的理解:类的对象只有有限个,确定的。我们称此为枚举类。 2.当需要定义一组常量时,强烈建议使用枚举类。对象便是所指的常量。 3.如果枚举类中只有一个对象,则可以作为单例模式的实现方式。 定义枚举类…...
基于java早餐店点餐系统源码设计与实现
摘 要 多姿多彩的世界带来了美好的生活,行业的发展也是形形色色的离不开技术的发展。作为时代进步的发展方面,信息技术至始至终都是成就行业发展的重要秘密。不论何种行业,大到国家、企业,小到团体、个人都在多方位的结合信息化技…...

ODOO16如何处理采购运输正常损耗的成本价核算?
《会计准则》规定:商品流通企业在采购商品过程中发生的运输费、装卸费、运输途中的合理损耗都归为采购存货成本中。 例如:采购A产品1000个,单价10元/个,途中运输正常损耗率是5%,因此实际入库是950个,入库金…...

【数据预测】基于白鲸优化算法BWO的VMD-KELM光伏发电功率预测 短期功率预测【Matlab代码#54】
文章目录 【可更换其他算法,获取资源请见文章第6节:资源获取】1. 白鲸优化算法BWO2. 变分模态分解VMD3. 核极限学习机KELM4. 部分代码展示5. 仿真结果展示6. 资源获取 【可更换其他算法,获取资源请见文章第6节:资源获取】 1. 白鲸…...
函数式编程-将过程作为返回值的应用:分步过程
之前的文章提到函数式编程的一等函数(First-class Function)四个性质中有“可以将过程作为返回值”这一点,但这一点在实际使用中不如“将过程作为参数”(高阶函数)用得多。本文介绍一种这个性质用于分步函数的应用。 …...

Mysql-学习笔记
文章目录 1. 数据库1.1 Mysql安装及常用代码1.2 SQL介绍1.3 SQL分类1. DDL-操作数据库,表2. DML-对表中的数据进行增删改3. DQL-对表中的数据进行查询条件查询模糊查询排序查询分组查询分页查询 4. DCL-对数据库进行权限控制外键约束表关系-多对多多表查询事务 1. 数…...

【雕爷学编程】Arduino动手做(187)---1.3寸OLED液晶屏模块2
37款传感器与模块的提法,在网络上广泛流传,其实Arduino能够兼容的传感器模块肯定是不止37种的。鉴于本人手头积累了一些传感器和执行器模块,依照实践出真知(一定要动手做)的理念,以学习和交流为目的&#x…...

Windows用户如何安装新版本cpolar内网穿透
Windows用户如何安装新版本cpolar内网穿透 文章目录 Windows用户如何安装新版本cpolar内网穿透 在科学技术高度发达的今天,我们身边充斥着各种电子产品,这些电子产品不仅为我们的工作带来极大的便利,也让生活变得丰富多彩。我们可以使用便携的…...
MacBookPro安装Win10,Wifi不能用了,触控板不能用了(2)
一、问题 去年在MacBookPro上装过Win10,当初只分配了60G空间。各方面原因需要重装系统,上个月装了一晚上,也无法连接Wifi,触控板只能当鼠标左键用。 后来发现是没有相关驱动造成的,于是从Mac系统联网找到网卡驱动&am…...

Linux应用开发之网络套接字编程(实例篇)
服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …...
Python爬虫实战:研究feedparser库相关技术
1. 引言 1.1 研究背景与意义 在当今信息爆炸的时代,互联网上存在着海量的信息资源。RSS(Really Simple Syndication)作为一种标准化的信息聚合技术,被广泛用于网站内容的发布和订阅。通过 RSS,用户可以方便地获取网站更新的内容,而无需频繁访问各个网站。 然而,互联网…...

Keil 中设置 STM32 Flash 和 RAM 地址详解
文章目录 Keil 中设置 STM32 Flash 和 RAM 地址详解一、Flash 和 RAM 配置界面(Target 选项卡)1. IROM1(用于配置 Flash)2. IRAM1(用于配置 RAM)二、链接器设置界面(Linker 选项卡)1. 勾选“Use Memory Layout from Target Dialog”2. 查看链接器参数(如果没有勾选上面…...

BCS 2025|百度副总裁陈洋:智能体在安全领域的应用实践
6月5日,2025全球数字经济大会数字安全主论坛暨北京网络安全大会在国家会议中心隆重开幕。百度副总裁陈洋受邀出席,并作《智能体在安全领域的应用实践》主题演讲,分享了在智能体在安全领域的突破性实践。他指出,百度通过将安全能力…...
Rapidio门铃消息FIFO溢出机制
关于RapidIO门铃消息FIFO的溢出机制及其与中断抖动的关系,以下是深入解析: 门铃FIFO溢出的本质 在RapidIO系统中,门铃消息FIFO是硬件控制器内部的缓冲区,用于临时存储接收到的门铃消息(Doorbell Message)。…...

2025年渗透测试面试题总结-腾讯[实习]科恩实验室-安全工程师(题目+回答)
安全领域各种资源,学习文档,以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具,欢迎关注。 目录 腾讯[实习]科恩实验室-安全工程师 一、网络与协议 1. TCP三次握手 2. SYN扫描原理 3. HTTPS证书机制 二…...
LRU 缓存机制详解与实现(Java版) + 力扣解决
📌 LRU 缓存机制详解与实现(Java版) 一、📖 问题背景 在日常开发中,我们经常会使用 缓存(Cache) 来提升性能。但由于内存有限,缓存不可能无限增长,于是需要策略决定&am…...

计算机基础知识解析:从应用到架构的全面拆解
目录 前言 1、 计算机的应用领域:无处不在的数字助手 2、 计算机的进化史:从算盘到量子计算 3、计算机的分类:不止 “台式机和笔记本” 4、计算机的组件:硬件与软件的协同 4.1 硬件:五大核心部件 4.2 软件&#…...
绕过 Xcode?使用 Appuploader和主流工具实现 iOS 上架自动化
iOS 应用的发布流程一直是开发链路中最“苹果味”的环节:强依赖 Xcode、必须使用 macOS、各种证书和描述文件配置……对很多跨平台开发者来说,这一套流程并不友好。 特别是当你的项目主要在 Windows 或 Linux 下开发(例如 Flutter、React Na…...

从零开始了解数据采集(二十八)——制造业数字孪生
近年来,我国的工业领域正经历一场前所未有的数字化变革,从“双碳目标”到工业互联网平台的推广,国家政策和市场需求共同推动了制造业的升级。在这场变革中,数字孪生技术成为备受关注的关键工具,它不仅让企业“看见”设…...