【数据结构】队列的完整实现
队列的完整实现
- 队列的完整实现
- github地址
- 前言
- 1. 队列的概念及其结构
- 1.1 概念
- 1.2 组织结构
- 2. 队列的实现
- 接口一览
- 结构定义与架构
- 初始化和销毁
- 入队和出队
- 取队头队尾数据
- 获取size和判空
- 完整代码与功能测试
- 结语
队列的完整实现
github地址
有梦想的电信狗
前言
队列(Queue)
作为一种基础且重要的数据结构,在计算机科学中扮演着关键角色。无论是操作系统的任务调度、网络数据包的管理,还是算法中的广度优先搜索(BFS)
,队列的“先进先出”(FIFO)
特性都使其成为不可或缺的工具。理解队列的实现原理,不仅能帮助开发者更高效地处理数据,还能为后续学习复杂的数据结构打下坚实基础。
本文将以 链式结构 为核心,详细介绍队列的完整实现。从结构设计、接口定义到功能测试,一步步剖析如何用C语言实现一个高效、健壮的队列。文章重点讲解入队(push
)、出队(pop
)、获取队头/队尾元素等核心操作,并通过清晰的代码示例和测试案例,帮助读者深入理解队列的内部机制。此外,所有代码已在GitHub
开源,方便读者参考和扩展。
1. 队列的概念及其结构
1.1 概念
队列:只允许在一端进行插入数据操作,在另一端进行删除数据操作的特殊线性表,队列具有先进先出FIFO(First In First Out)
的特性
- 入队列:进行插入操作的一端称为队尾,入队常被称为
push
- 出队列:进行删除操作的一端称为队头,出队常被称为
pop
- 如下图所示
1.2 组织结构
-
队列可以使用数组和链表的结构实现,使用链表的结构实现更优一些。
-
因为如果使用数组的结构,出队列时,在数组头上出数据,效率会比较低
-
而对于链表实现的队列来说,入队对应尾插操作,出队对应头删操作。在链表中,头删和尾插操作只需要
O(1)
时间复杂度,因此队列更适合使用链表来实现,本文我们采用单链表来实现。
2. 队列的实现
接口一览
//初始化 与 销毁队列
void QueueInit(Queue* pQueue);
void QueueDestroy(Queue* pQueue);// 队尾入队列 队头出队列
void QueuePush(Queue* pQueue, QDataType data);
void QueuePop(Queue* pQueue);// 获取队列头部元素 /获取队列尾部元素
QDataType QueueFront(Queue* pQueue);
QDataType QueueBack(Queue* pQueue);//获取队列中有效元素个数 检测队列是否为空,如果为空返回非零结果,如果非空返回0
int QueueSize(Queue* pQueue);
bool QueueEmpty(Queue* pQueue);
结构定义与架构
结构:
//队列,先进先出,数组的话在头部出数据不方便,因此用链表来实现,单链表
typedef int QDataType;typedef struct QNode { //链式队列,用单链表实现struct QNode* next;QDataType data;
}QNode;//队列中 用两个指针来指示 队头和队尾,方便入队和出队
typedef struct Queue {QNode* head;QNode* tail;int size;
}Queue;
- 使用
typedef int QDataType
方便队列中存放不同的数据类型 QNode
表示我们链表中一个个的结点,内部包含next指针
和数据data
- 使用
struct Queue
结构来表示整个队列,其中:- 规定两个
QNode*
的指针,分别保存链表第一个结点和尾结点的地址(分别指向第一个结点和最后一个结点) - 定义
int size
来保存队列中的有效元素个数。
- 规定两个
架构图如下:
初始化和销毁
初始化:
//初始化 与 销毁队列
void QueueInit(Queue* pQueue) {assert(pQueue);pQueue->head = pQueue->tail = NULL;pQueue->size = 0;
}
- 通过
Queue
结构体的指针pQueue
,来访问结构体中的成员head
和tail
,通过head
和tail
指针和链表的特性,可以访问到链表中的每个结点。head
和tail
指针主要是为了方便访问队头结点和队尾结点。 assert(pQueue)
保证Queue
结构存在- 初始化队列:
- 链表中无节点时,
head
和tail
指针都置为NULL
- 初始化
size
为0
- 链表中无节点时,
销毁:
//销毁
void QueueDestroy(Queue* pQueue) {assert(pQueue);// 空链表也可以销毁,因此无需断言链表非空QNode* cur = pQueue->head;while (cur != NULL) {QNode* next = cur->next;free(cur);cur = next;}pQueue->head = pQueue->tail = NULL;pQueue->size = 0;
}
assert(pQueue)
保证Queue
结构存在QNode* cur = pQueue->head
:cur
保存当前头结点的地址while
循环依次释放每一个结点- 保存
cur
的下一个节点cur->next
free(cur)
释放当前结点,cur
移动指向下一个节点,直到NULL
时结束
pQueue->head = pQueue->tail = NULL
:将head
和tail
指针各自置NULL
pQueue->size = 0
最后将size
置0
入队和出队
入队:
// 队尾入队列 队头出队列
void QueuePush(Queue* pQueue, QDataType data) {assert(pQueue);// 开辟新结点,并将数据置于新结点中QNode* newNode = (QNode*)malloc(sizeof(QNode));if (newNode == NULL) {perror("malloc failed\n");return;}newNode->data = data;newNode->next = NULL;// 初始化后,head 和 tail 都为NULLif (pQueue->head == NULL) {assert(pQueue->tail == NULL); // head为NULL时,tail必须也为NULLpQueue->head = pQueue->tail = newNode;}else {pQueue->tail->next = newNode;pQueue->tail = newNode;}pQueue->size++;
}
assert(pQueue)
保证Queue
结构存在QNode* newNode = (QNode*)malloc(sizeof(QNode)); if (newNode == NULL) { perror("malloc failed\n"); return; }
开辟一个新结点,并进行初始化。- 初始化
QNode
后,push
一个结点本质是单链表的尾插,有两种情况:- 队列为空(队列中无节点)时:新开辟的
QNode
应当成为链表中的第一个节点,pQueue->head = pQueue->tail = newNode
操作调整首尾指针即可。 - 队列中已有其他结点时:此时是单链表的尾插。
pQueue->tail->next = newNode
:新结点链入原链表pQueue->tail = newNode
:更改尾指针指向
- 队列为空(队列中无节点)时:新开辟的
push
过后,pQueue->size++
,队列的size
应当++
出队:
v1版本仅实现:
void QueuePop(Queue* pQueue) {assert(pQueue);assert(pQueue->head != NULL);//if(pQueue->head->next == NULL){} //考虑只剩一个结点的情况if (pQueue->head == pQueue->tail) { free(pQueue->head);pQueue->head = pQueue->tail = NULL;}else {QNode* cur = pQueue->head;pQueue->head = pQueue->head->next;free(cur);cur = NULL;}pQueue->size--;
}
assert(pQueue)
:断言队列存在,assert(pQueue->head != NULL)
,确保pop
时队列内有元素pop
出队时,销毁第一个结点,确保队列存在且有元素后,此处存在两种情况- 当前仅剩一个结点时:
pQueue->head == pQueue->tail
,直接free
当前头结点,将head
和tail
指针置NULL
即可 - 当前存在多个节点时(else):
QNode* cur = pQueue->head
:记录下当前的头结点pQueue->head = pQueue->head->next
:头指针向后移动free(cur)
后cur = NULL
释放之前的头结点的空间,并将cur
置NULL
- 当前仅剩一个结点时:
pop
后size
应当--
v2版本优化:
- 可以看到,上述的代码中存在多次
free
,且都是对要被删除的结点进行free
,那么是否可以优化为一次free
呢
void QueuePop(Queue* pQueue) {assert(pQueue);assert(pQueue->head != NULL);//优化版本QNode* cur = pQueue->head;if (pQueue->head == pQueue->tail)pQueue->head = pQueue->tail = NULL;elsepQueue->head = pQueue->head->next;free(cur);cur = NULL;pQueue->size--;
}
assert(pQueue)
:断言队列存在,assert(pQueue->head != NULL)
,确保pop
时队列内有元素QNode* cur = pQueue->head
:不管队列中剩余几个结点,最终都要free
头结点,那么cur
直接保存头结点的地址,方便进行操作pQueue->head == pQueue->tail仅剩一个结点时
:仅需对head
和tail
指针做修改。有多个节点时,pQueue->head = pQueue->head->next
,head
指针向后移动。- 最终
cur
内保存了要被删除的结点的地址,直接free(cur)
并置NULL
pop
后size
应当--
取队头队尾数据
获取队头数据:
// 获取队列头部元素 获取队列尾部元素
QDataType QueueFront(Queue* pQueue) {assert(pQueue);//确保队列非空assert(!QueueEmpty(pQueue));return pQueue->head->data;
}
assert(pQueue)
确保队列存在,assert(!QueueEmpty(pQueue))
确保队列非空,非空队列内才有数据- 头部数据:
return pQueue->head->data
,通过头指针head
访问队列内第一个结点的数据
获取队尾数据:
//获取队列尾部元素
QDataType QueueBack(Queue* pQueue) {assert(pQueue);//确保队列非空assert(!QueueEmpty(pQueue));return pQueue->tail->data;
}
assert(pQueue)
确保队列存在,assert(!QueueEmpty(pQueue))
确保队列非空,非空队列内才有数据- 尾部数据:
return pQueue->tail->data
,通过尾指针tail
访问队列内最后一个结点的数据
获取size和判空
获取size:
int QueueSize(Queue* pQueue) {assert(pQueue);return pQueue->size;
}
assert(pQueue)
:断言指针非空,确保Queue
结构体存在- 直接返回
size
即可得到元素个数- 这里就体现出我们在
Queue
结构中添加一个size
成员的好处,只需在每次Push/Pop
后对size
进行加减,即可方便的得到队列的size
- 如果
Queue
结构内不维护一个size
变量的话,由于我们的队列是基于单链表实现的,每次获取队列的大小时都只能遍历链表来得到size
,遍历的时间复杂度较高。
- 这里就体现出我们在
判空:
bool QueueEmpty(Queue* pQueue) {assert(pQueue);//return (pQueue->head == NULL && pQueue->tail == NULL);return pQueue->size == 0; // size为0时为空
}
assert(pQueue)
:断言指针非空,确保Queue
结构体存在pQueue->size == 0
或pQueue->head == NULL && pQueue->tail == NULL
,两个条件均可以标识队列为空
完整代码与功能测试
完整代码如下:
#pragma once
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
#include <stdbool.h>//队列,先进先出,数组的话在头部出数据不方便,因此用链表来实现,单链表
typedef int QDataType;typedef struct QNode { //链式队列,用单链表实现struct QNode* next;QDataType data;
}QNode;//队列中 用两个指针来指示 队头和队尾,方便入队和出队
typedef struct Queue {QNode* head;QNode* tail;int size;
}Queue;//初始化 与 销毁队列
void QueueInit(Queue* pQueue);
void QueueDestroy(Queue* pQueue);// 队尾入队列 队头出队列
void QueuePush(Queue* pQueue, QDataType data);
void QueuePop(Queue* pQueue);// 获取队列头部元素 /获取队列尾部元素
QDataType QueueFront(Queue* pQueue);
QDataType QueueBack(Queue* pQueue);//获取队列中有效元素个数 检测队列是否为空,如果为空返回非零结果,如果非空返回0
int QueueSize(Queue* pQueue);
bool QueueEmpty(Queue* pQueue);// #include "Queue.h" 多文件编译时,需正确包含头文件//初始化 与 销毁队列
void QueueInit(Queue* pQueue) {assert(pQueue);pQueue->head = pQueue->tail = NULL;pQueue->size = 0;
}
//销毁
void QueueDestroy(Queue* pQueue) {assert(pQueue);QNode* cur = pQueue->head;while (cur != NULL) {QNode* next = cur->next;free(cur);cur = next;}pQueue->head = pQueue->tail = NULL;pQueue->size = 0;
}// 队尾入队列 队头出队列
void QueuePush(Queue* pQueue, QDataType data) {assert(pQueue);QNode* newNode = (QNode*)malloc(sizeof(QNode));if (newNode == NULL) {perror("malloc failed\n");return;}newNode->data = data;newNode->next = NULL;// 初始化后,head 和 tail 都为NULLif (pQueue->head == NULL) {assert(pQueue->tail == NULL); // head为NULL时,tail必须也为NULLpQueue->head = pQueue->tail = newNode;}else {pQueue->tail->next = newNode;pQueue->tail = newNode;}pQueue->size++;
}
//队列的头删法
void QueuePop(Queue* pQueue) {assert(pQueue);assert(pQueue->head != NULL);if(pQueue->head->next == NULL){} //考虑只剩一个结点的情况//if (pQueue->head == pQueue->tail) { // free(pQueue->head);// pQueue->head = pQueue->tail = NULL;//}//else {// QNode* cur = pQueue->head;// pQueue->head = pQueue->head->next;// free(cur);// cur = NULL;//}//pQueue->size--;//优化版本QNode* cur = pQueue->head;if (pQueue->head == pQueue->tail)pQueue->head = pQueue->tail = NULL;elsepQueue->head = pQueue->head->next;free(cur);cur = NULL;pQueue->size--;
}// 获取队列头部元素 获取队列尾部元素
QDataType QueueFront(Queue* pQueue) {assert(pQueue);//确保队列非空assert(!QueueEmpty(pQueue));return pQueue->head->data;
}
//获取队列尾部元素
QDataType QueueBack(Queue* pQueue) {assert(pQueue);//确保队列非空assert(!QueueEmpty(pQueue));return pQueue->tail->data;
}//获取队列中有效元素个数 检测队列是否为空,如果为空返回非零结果,如果非空返回0
// 双向链表中,不能用 哨兵位的数据 来存储 链表的长度
// 之前的实现中,结点内存放的数据是int,导致哨兵位内的数据位类型也是int,
int QueueSize(Queue* pQueue) {assert(pQueue);return pQueue->size;
}
bool QueueEmpty(Queue* pQueue) {assert(pQueue);//return (pQueue->head == NULL && pQueue->tail == NULL);return pQueue->size == 0; // size为0时为空
}
功能测试:
// 需正确包含头文件
#include "Queue.h"void TestQueue() {Queue queue;QueueInit(&queue);QueuePush(&queue, 1);QueuePush(&queue, 2);QueuePush(&queue, 3);QueuePush(&queue, 4);QueuePush(&queue, 6);// 遍历的代码/*while (!QueueEmpty(&queue)) {printf("%d ", QueueFront(&queue));QueuePop(&queue);}*/printf("队尾:%d 有效元素个数:%d\n", QueueBack(&queue), QueueSize(&queue));printf("队头:%d 有效元素个数:%d\n", QueueFront(&queue), QueueSize(&queue));QueuePop(&queue);printf("队尾:%d 有效元素个数:%d\n", QueueBack(&queue), QueueSize(&queue));printf("队头:%d 有效元素个数:%d\n", QueueFront(&queue), QueueSize(&queue));QueuePop(&queue);printf("队尾:%d 有效元素个数:%d\n", QueueBack(&queue), QueueSize(&queue));printf("队头:%d 有效元素个数:%d\n", QueueFront(&queue), QueueSize(&queue));QueuePop(&queue);QueuePop(&queue);printf("队列有效元素个数:%d\n", QueueSize(&queue));while (!QueueEmpty(&queue)) {printf("%d ", QueueFront(&queue));QueuePop(&queue);}QueueDestroy(&queue);
}//函数调用的栈帧 和 数据结构的栈
// 函数调用的栈帧 是操作系统层面对内存区域的划分
// 和 数据结构的栈int main() {TestQueue();return 0;
}
结语
通过本文的学习,我们完成了队列的完整实现:从结构体设计到核心接口的实现,再到功能验证。关键点包括:
- 链式结构的优势:使用单链表实现队列,确保入队(尾插)和出队(头删)操作的时间复杂度均为
O(1)
。 - 双指针的妙用:通过
head
和tail
指针分别标记队头和队尾,避免了遍历链表的性能损耗。 - 健壮性保障:通过断言(
assert
)确保操作合法性,并结合size
字段快速判断队列状态。
希望本文能帮助你彻底掌握队列的实现原理,并激发对数据结构更深层次的探索。
以上就是本文的所有内容了,如果觉得文章对你有帮助,欢迎 点赞⭐收藏 支持!如有疑问或建议,请在评论区留言交流,我们一起进步
分享到此结束啦
一键三连,好运连连!
你的每一次互动,都是对作者最大的鼓励!
征程尚未结束,让我们在广阔的世界里继续前行!
🚀
相关文章:
【数据结构】队列的完整实现
队列的完整实现 队列的完整实现github地址前言1. 队列的概念及其结构1.1 概念1.2 组织结构 2. 队列的实现接口一览结构定义与架构初始化和销毁入队和出队取队头队尾数据获取size和判空 完整代码与功能测试结语 队列的完整实现 github地址 有梦想的电信狗 前言 队列&…...
销售易史彦泽:从效率工具到增长引擎,AI加速CRM不断进化
导读:AI的加入,让CRM实现从“人适配系统”到“系统适配人”,从“管控工具”向“智能助手”跃迁,重构客户关系管理的底层逻辑。 作者 | 小葳 图片来源 | 摄图 AI应用与SaaS的关系,是当前科技与商业领域热议的话题。 当…...
开疆智能Profinet转ModbusTCP网关连接BORUNTE伯朗特系统配置案例
本案例是通过开疆智能Profinet转ModbusTCP网关将西门子PLC与BORUNTE机器人连接的配置案例。具体配置方法如下。 配置过程 Profinet设置 设置网关在Profinet一侧的参数包括(设备名称,IP地址等) 先导入GSD文件 选择GSD所在文件夹位置&#…...
从0到1搭建shopee测评自养号系统:独立IP+硬件伪装+养号周期管理
在跨境电商竞争白热化的背景下,Shopee卖家通过自养号测评实现流量与销量突破已成为行业共识。自养号测评通过模拟真实买家行为,为店铺注入精准流量,同时规避外包测评的高风险与不可控性。本文将从技术架构、运营策略、风险控制三个维度&#…...
arrow-0.1.0.jar 使用教程 - Java jar包运行方法 命令行启动步骤 常见问题解决
准备工作 首先确保你电脑上装了Java环境(JDK 8或以上版本) 把这个jar文件下载到你的电脑上,arrow-0.1.0.jar下载链接:https://pan.quark.cn/s/66d7c061c95a 运行方法 打开命令行(Windows按WinR输入cmd,M…...
请问交换机和路由器的区别?vlan 和 VPN 是什么?
交换机和路由器的区别 特性交换机(Switch)路由器(Router)工作层级数据链路层(L2,基于MAC地址)网络层(L3,基于IP地址)主要功能在局域网(LAN&#…...
如何查看与设置电脑静态IP地址:完整指南
在当今数字化时代,稳定的网络连接已成为工作生活的必需品。静态IP地址作为网络配置中的重要一环,相比动态IP具有更高的稳定性和可控性,然而,许多用户对如何查看和设置静态IP地址仍感到困惑。本文将为您提供从基础概念到实操步骤的…...
Linux网络基础全面解析:从协议分层到局域网通信原理
Linux系列 文章目录 Linux系列前言一、计算机网络背景1.1 认识网络1.2 认识协议 二、网络协议初识2.1 协议分层2.2 OSI七层模型2.3 TCP/IP协议栈2.4 网络协议栈与OS的关系2.5 网络协议在网络传输时的作用 三、网络通信局域网通信的安全隐患与应对总结 前言 Linux系统部分的学习…...
第二篇:服务与需求——让用户找到并预订服务
目录 1 服务类目与项目管理:飞书多维表格为管理中心,微搭小程序展示1.1 需求分析1.2 数据模型:微搭中的服务分类与服务项目(用于小程序展示)1.3 数据模型:多维表格中的服务分类与服务项目 总结 我们已经用了…...
【AI News | 20250520】每日AI进展
AI Repos 1、nanoDeepResearch nanoDeepResearch 是一个受 ByteDance 的 DeerFlow 项目启发,旨在从零开始构建深度研究代理的后端项目。它不依赖 LangGraph 等现有框架,通过实现一个 ReAct 代理和状态机来模拟 Deep Research 的工作流程。项目主要包含规…...
Spark Core基础与源码剖析全景手册
Spark Core基础与源码剖析全景手册 Spark作为大数据领域的明星计算引擎,其核心原理、源码实现与调优方法一直是面试和实战中的高频考点。本文将系统梳理Spark Core与Hadoop生态的关系、经典案例、聚合与分区优化、算子底层原理、集群架构和源码剖析,结合…...
抖音视频如何下载保存?高清无水印一键保存到手机!
你是不是经常在抖音上刷到超有趣的短视频,想保存下来却不知道怎么做?或者下载后发现带有烦人的水印?别担心!今天教你最简单、最快速的抖音视频下载方法,无水印、高清画质,轻松搞定! 为什么要下…...
SCAU--平衡树
3 平衡树 Time Limit:1000MS Memory Limit:65535K 题型: 编程题 语言: G;GCC;VC;JAVA;PYTHON 描述 平衡树并不是平衡二叉排序树。 这里的平衡指的是左右子树的权值和差距尽可能的小。 给出n个结点二叉树的中序序列w[1],w[2],…,w[n],请构造平衡树,…...
图的几种存储方法比较:二维矩阵、邻接表与链式前向星
图是一种非常重要的非线性数据结构,广泛应用于社交网络、路径规划、网络拓扑等领域。在计算机中表示和存储图结构有多种方法,本文将详细分析三种常见的存储方式:二维矩阵(邻接矩阵)、邻接表和链式前向星,比…...
【AS32X601驱动系列教程】MCU启动详解
在嵌入式开发领域,掌握MCU(微控制单元)的启动流程是工程师们迈向深入开发的关键一步。本文将带您深入了解MCU启动的奥秘,从编译过程到启动文件,再到链接脚本和系统时钟配置,全方位解析MCU启动流程。 在实际…...
计算机视觉与深度学习 | Matlab实现EMD-GWO-SVR、EMD-SVR、GWO-SVR、SVR时间序列预测(完整源码和数据)
以下是一个完整的Matlab时间序列预测实现方案,包含EMD-GWO-SVR、EMD-SVR、GWO-SVR和SVR四种方法的对比。代码包含数据生成、信号分解、优化算法和预测模型实现。 %% 主程序:时间序列预测对比实验 clc; clear; clearvars; close all;% 生成模拟时间序列数据 rng(1); % 固定随…...
Visual Studio 2022 插件推荐
Visual Studio 2022 插件推荐 Visual Studio 2022 (简称 VS2022) 是一款强大的 IDE,适合各类系统组件、框架和应用的开发。插件是接入 VS2022 最重要的扩展方式之一,它们可以大幅提升开发效率、优化代码质量,并提供强大的调试和分析功能。 …...
[luogu12541] [APIO2025] Hack! - 交互 - 构造 - 数论 - BSGS
传送门:https://www.luogu.com.cn/problem/P12541 题目大意:有一个数 n n n,你不知道是多少;你每次可以向交互库询问一个正整数集合 A A A(其中元素互不相同),交互库返回:将集合中…...
openjdk底层(hotspot)汇编指令调用(五)——内存访问
根据前面关于aarch64架构下的编码解释可知,在src\hotspot\cpu\架构文件夹下, assembler_xx.hpp assembler_xx.cpp register_xx.hpp register_xx.cpp register_definitions_xx.cpp这些文件是有关寄存器定义以及汇编编码函数实现的文件。 对于前述的ope…...
几款常用的虚拟串口模拟器
几款常用的虚拟串口模拟器(Virtual Serial Port Emulator),适用于 Windows 系统,可用于开发和调试串口通信应用: 1. com0com (开源免费) 特点: 完全开源免费,无功能限制。 可创建多个虚拟串口…...
ChimeraX介绍
UCSF ChimeraX 是一款由美国加州大学旧金山分校(UCSF)开发的下一代分子可视化软件,是经典的 UCSF Chimera 的继任者。它集成了强大的分子结构可视化、分析、建模和动画功能,广泛应用于结构生物学、药物设计、分子建模等领域。 1. 下载安装: Download UCSF ChimeraX 2. …...
【Linux】初见,基础指令
前言 本文将讲解Linux中最基础的东西-----指令,带大家了解一下Linux中有哪些基础指令,分别有什么作用。 本文中的指令和选项并不全,只介绍较为常用的 pwd指令 语法:pwd 功能:显示当前所在位置(路径…...
链表的面试题8之环形链表
许久不见,那么这是最后倒数第三题了,这道题我们来看一下环形链表。 老规矩贴链接:141. 环形链表 - 力扣(LeetCode) 目录 倒数第k个元素 获取中间元素的问题。 双指针 来,大致看一下题目,这…...
OBS Studio:windows免费开源的直播与录屏软件
OBS Studio是一款免费、开源且跨平台的直播与录屏软件。其支持 Windows、macOS 和 Linux。OBS适用于,有直播需求的人群或录屏需求的人群。 Stars 数64,323Forks 数8413 主要特点 推流:OBS Studio 支持将视频实时推流至多个平台,如 YouTube、…...
邂逅Node.js
首先先要来学习一下nodejs的基础(和后端开发有联系的) 再然后的学习路线是学习npm,yarn,cnpm,npx,pnpm等包管理工具 然后进行模块化的使用,再去学习webpack和git(版本控制工具&…...
React 常见的陷阱之(如异步访问事件对象)
文章目录 前言1. 异步访问事件对象问题解决方案 2. 事件传播的误解**问题**解决方案 **3. 事件监听器未正确卸载****问题****解决方案** **4. 动态列表中的事件绑定****问题****解决方案** **5. 第三方库与 React 事件冲突****问题****解决方案** **6. 表单输入与受控组件****问…...
【LinkedList demo 内部类讲说】
LinkedList demo 内部类讲说 1. Node节点2.MyLinkedList3. LinkedListTest 测试类 1. Node节点 public class Node<T> {private Node<T> pre;private Node<T> next;private T data;public Node() {}public Node getPre() {return pre;}public void setPre(N…...
Sql刷题日志(day9)
一、笔试 1、limit offset:分页查询 SELECT column1, column2, ... FROM table_name LIMIT number_of_rows OFFSET start_row; --跳过前 start_row 行,返回接下来的 number_of_rows 行。 2、lag、lead:查询前后行数据 --lag函数用于访问当…...
46 python pandas
Pandas是Python数据分析的利器,也是各种数据建模的标准工具 一、什么是pandas pandas 是 Python 中用于数据处理和分析的核心库,提供了高效的数据结构(如Series和DataFrame)和数据操作工具,广泛应用于数据清洗、分析、可视化等场景。 最常用的是用来处理excel数据。 二…...
告别延迟!Ethernetip转modbustcp网关在熔炼车间监控的极速时代
熔炼车间热火朝天,巨大的热风炉发出隆隆的轰鸣声,我作为一名技术操控工,正密切关注着监控系统上跳动的各项参数。这套基于EtherNET/ip的监控系统,是我们车间数字化改造的核心,它将原本分散的控制单元整合在一起&#x…...
Prompt Tuning:高效微调大模型的新利器
Prompt Tuning(提示调优)是什么 Prompt Tuning(提示调优) 是大模型参数高效微调(Parameter-Efficient Fine-Tuning, PEFT)的重要技术之一,其核心思想是通过优化 连续的提示向量(而非整个模型参数)来适配特定任务。以下是关于 Prompt Tuning 的详细解析: 一、核心概念…...
⼆叉搜索树详解
1. ⼆叉搜索树的概念 ⼆叉搜索树⼜称⼆叉排序树,它或者是⼀棵空树,或者是具有以下性质的⼆叉树: • 若它的左⼦树不为空,则左⼦树上所有结点的值都⼩于等于根结点的值 • 若它的右⼦树不为空,则右⼦树上所有结点的值都⼤于等于根结…...
CompleteableFuture的异步任务编排
为什么会有CompleteableFuture Java 的 1.5 版本引入了 Future,可以把它简单的理解为运算结果的占位符, 它提供了两个方法来获取运算结果。 get():调用该方法线程将会无限期等待运算结果。get(longmeout, TimeUnit unit):调用该…...
珈和科技贺李德仁院士荣膺国际数字地球学会会士:以时空智能赋能可持续发展目标 绘就数字地球未来蓝图
4月22日,第十四届国际数字地球会议在重庆盛大启幕。在这场在全球范围内数字地球领域具有国际影响力的学术盛会上,国际数字地球学会向珈和科技的企业顾问,2023年度国家最高科学技术奖得主李德仁院士授予了“国际数字地球学会会士”最高荣誉称号…...
【CodeBuddy 】从0到1,打造一个“牛马打鸡血仪”
【CodeBuddy 】从0到1,打造一个“牛马打鸡血仪” 我正在参加CodeBuddy「首席试玩官」内容创作大赛,本文所使用的 CodeBuddy 免费下载链接:腾讯云代码助手 CodeBuddy - AI 时代的智能编程伙伴 🌟嗨,我是LucianaiB&#…...
BI是什么意思?一文讲清BI的概念与应用!
目录 一、BI 是什么意思 1. BI 的定义 2. BI 的发展历程 3. BI 的核心组件 二、BI 的应用场景 1. 销售与市场营销 2. 财务管理 编辑3. 人力资源管理 4. 生产与运营管理 编辑三、选择合适的 BI 工具 1. 考虑企业的需求和规模 2. 评估工具的功能和性能 3. 关注工…...
可编辑PPT | 华为安全架构设计方法指南华为数字化转型架构解决方案
这份文档是华为的安全架构设计方法指南,它详细介绍了安全架构设计的重要性、方法和流程。文档强调安全架构是软件研发技术体系中的关键DFX能力,与可靠性、性能等并列,尤其在云计算和复杂网络环境下,安全性设计显得尤为重要。华为的…...
1.6 提示词工程(二)
目录 3.2 提供参考文本 3.2.1 使用参考文本来构建答案 3.2.2 指导模型用引用的文本回答问题 3.3 把复杂的任务拆分成简单的子任务 3.3.1 利用意图分类确定与用户查询最相关的指令 3.3.2 针对需要长时间对话的应用程序,应概括或过滤之前的对话内容 …...
WIFI信号状态信息 CSI 深度学习之数据集
Building occupant activity sensing dataset based on WIFI CSI(WiSA) 所有的数据以及实验参数都上传到了figshare中并配备详细说明,供参考。 论文链接:WiSA: Privacy-enhanced WiFi-based activity intensity recognition in …...
基于服务器的 DPI 深度分析解决方案
一、传统网络流量分析的瓶颈与挑战 在企业网络管理体系中,传统流量分析模式高度依赖网络设备作为数据采集核心节点,无论是基于 NetFlow/IPFIX 等流协议的流量分析,还是通过端口镜像技术实现的流量监控,均以交换机、路由器等网络设…...
动态规划(5):线性动态规划
引言 所谓线性动态规划,通常指状态定义和转移具有线性结构的动态规划问题,其状态通常可以用一维数组表示,状态转移主要依赖于相邻或前面有限个状态。这类问题的特点是状态空间呈线性排列,每个状态只与有限个前置状态相关,使得问题结构相对简单,更容易理解和掌握。 一维…...
c语言- 如何构建CMake项目(Linux/VSCode)
目录 linux(vscode)构建C语言CMake项目 1. 检查linux是否下载cmake,否则执行下列代码 2. 在vscode下载cmake的插件CMake Tools 3. 构建项目(项目结构) 4. 进行cmake配置 1. 在VS Code中按下ctrl shift p键&…...
HJ17 坐标移动【牛客网】
文章目录 零、原题链接一、题目描述二、测试用例三、解题思路四、参考代码 零、原题链接 HJ17 坐标移动 一、题目描述 二、测试用例 三、解题思路 基本思路: 这题的难点在于理解题目和如何处理各种情况。题目是给定一串指令,首先要判断指令是否合法…...
HGHAC集群滚动扩展或更换硬盘设备
文章目录 环境文档用途详细信息 环境 系统平台:N/A 版本:4.5.8 文档用途 集群版本:hghac4.2.1 数据库版本:hgdb-see-4.5.8 此步骤适用于所有hac架构的hgdb集群。 主要用途:HAC集群服务器滚动扩展或更换硬盘 本文…...
虚拟环境中VSCode运行jupyter文件
用VS Code打开jupyter文件,点击右上角 Select Kernel 在正上方会出现这个选择框,选择 Python Environment 会出来所有的虚拟环境,选择要用的环境行...
【蓝桥杯嵌入式】【模块】六、PWM相关配置及代码模板
1. 前言 最近在准备16届的蓝桥杯嵌入式赛道的国赛,打算出一个系列的博客,记录STM32G431RBT6这块比赛用板上所有模块可能涉及到的所有考点,如果有错误或者遗漏欢迎各位大佬斧正。 本系列博客会分为以下两大类: 1.1. 单独模块的讲…...
力扣-盛最多水的容器
1.题目描述 2.题目链接 11. 盛最多水的容器 - 力扣(LeetCode) 3.题目解析 题目中的储水量两边差*短边高度。也就是说,两条边中,决定储水量的是短边的高度。 我们可以定义两个指针,一个在最左边,一个在…...
数据实时同步:inotify + rsync 实现数据实时同步
1 数据实时同步 在生产环境中,某些场景下,要将数据或文件进行实时同步,保证数据更新后其它节点能立即获得最新的数据。 数据同步的两种方式 PULL:拉,使用定时任务的方式配合同步命令或脚本等,从指定服务…...
C#学习第24天:程序集和部署
程序集知识点 1.程序集的基本概念 程序集是部署和版本控制的最小单位。它可以是可执行文件(.exe)或动态链接库(.dll)。包含元数据和清单(Manifest),描述程序集的内容和依赖关系。 2.程序集清单…...
mac .zshrc:1: command not found: 0 解决方案
nano ~/.zshrc 使用自带的nano命令打开文件,修改后 Ctrl X 然后输入y 然后回车即可保存成功 一般情况下,不是常用这个命令,除非是遇到有问题的文件,才用, 例如 遇到下面的问题 /Users/xxli/.zshrc:1: command no…...