当前位置: 首页 > news >正文

数据结构(三)---单向循环链表

 单向循环链表(Circular Linked List)

一、基本概念

循环链表是一种特殊的链表,其末尾节点的后继指针指向头结点,形成一个闭环。

循环链表的操作与普通链表基本一致,但需注意循环特性的处理。

二、代码实现

clList.h
#ifndef _CLLIST_H
#define _CLLIST_H#include <stdio.h>
#include <stdlib.h>
#include <string.h>//定义节点数据的类型
typedef int DATA;//定义一个单向循环链表的节点
typedef struct node
{DATA data;     //节点数据域struct node *next;     //指针域,指向后继节点
}NODE;//函数原型的声明
/*** 创建链表* @param head 待操作的链表* @param data 待插入的数据* @return 成功返回0,否则返回-1*/
extern int cllist_create(NODE **head, DATA data);/*** 在循环链表插入新节点(把新节点插入到头节点之前)* @param head 待操作的链表(默认是头节点)* @param data 待插入的数据* @return 成功返回0,否则返回-1*/
extern int cllist_insert(NODE **head, DATA data);/*** 实现无头节点的头插法(插入在头节点之前)* @param head 待操作的链表(默认是头节点)* @param data 待插入的数据* @return 成功返回0,否则返回-1*/
extern int cllist_insertAthead(NODE **head, DATA data);/*** 遍历链表数据* @param head 待遍历的链表* @return 成功返回0,否则返回-1*/
extern int cllist_showAll(const NODE *head);/*** 根据data返回查找对应的节点* @param head 待操作的链表* @param data 需要查找的节点数据* @return 查找到的节点*/
extern NODE *cllist_find(const NODE *head, DATA data);/*** 根据newdata修改old对应的节点数据* @param head 待操作的链表* @param old 待修改的原数据* @param newdata 待修改的新数据* @return 成功返回0,否则返回-1*/
extern int cllist_update(const NODE *head, DATA old, DATA newdata);/*** 根据data删除对应的节点* @param head 待操作的链表* @param data 待删除的节点数据* @return 成功返回0,否则返回-1*/
extern int cllist_delete(NODE **head, DATA data);/*** 销毁整个链表* @param head 待销毁的链表*/
extern void cllist_destroy(NODE **head);#endif //_CLLIST_H
clList.c
#include "clList.h"//函数原型的声明
/*** 创建链表* @param head 待操作的链表(head是指向头节点指针的地址)* @param data 待插入的数据* @return 成功返回0,否则返回-1*/
int cllist_create(NODE **head, DATA data)
{//如果链表存在,就无需创建if(*head){//打印错误信息到标准错误流(避免影响正常输出)fprintf(stderr, "链表已存在,无需创建!\n");return -1;}//单向循环链表//创建一个新节点NODE *p = (NODE*)malloc(sizeof(NODE));//校验新节点是否创建成功if(!p)return -1;//初始化新节点p->data = data;p->next = p;     //循环特性:单个节点的next指向自身(即使单个节点也形成环)//设置头指针*head = p;     //将头指针设置为自己
}/*** 实现无头节点的头插法(插入在头节点之前)* @param head 待操作的链表(默认是头节点)* @param data 待插入的数据* @return 成功返回0,否则返回-1*/
int cllist_insertAthead(NODE **head, DATA data)
{//创建一个新节点NODE *pNew = (NODE*)malloc(sizeof(NODE));if(!pNew){fprintf(stderr, "创建新节点失败!\n");return -1;}//初始化新节点的数据域,指针域暂不赋值pNew->data = data;NODE *p = *head;     //创建一个遍历指针用来遍历链表寻找尾节点//如果是空链表if(!p){pNew->next = pNew;*head = pNew;return 0;}//非空链表pNew->next = *head;     //新节点的next指向原头节点//查找尾节点(原链表的最后一个节点)while(p->next != *head)     //单向循环链表的尾节点next永远指向头节点(非NULL){p = p->next;}//此时p是尾节点p->next = pNew;*head = pNew;return 0;
}/*** 在循环链表插入新节点(把新节点插入到头节点之前)* @param head 待操作的链表(默认是头节点)* @param data 待插入的数据* @return 成功返回0,否则返回-1*/
int cllist_insert(NODE **head, DATA data)
{//创建新节点NODE *p = (NODE*)malloc(sizeof(NODE));//校验节点是否创建成功if(!p)return -1;//初始化新节点p->data = data;p->next = p;     //此时创建的新节点与链表还无关//情景1:若待插入的链表是空链表if(*head == NULL){*head = p;     //设置头指针,相当于创建了一个新链表return 0;}//情景2:若待插入的链表是非空链表(在头节点和头节点的next之间插入,头节点不变)p->next = (*head)->next;     //新节点指向原头节点的下一个节点(*head)->next = p;     //头节点指向新节点}/*** 遍历链表数据* @param head 待遍历的链表* @return 成功返回0,否则返回-1*/
int cllist_showAll(const NODE *head)
{//创建临时指针用于遍历(保持头指针不变)const NODE *p = head;     //p指向当前遍历的节点//空链表if(!p){fprintf(stderr, "空链表,没有数据!\n");return -1;}//遍历列表do     //使用do—while保证至少执行依次(应对单节点循环链表){printf("%d\t", p->data);p = p->next;}while(p != head);     //循环终止条件:回到起始节点printf("\n");return 0;
}/*** 根据data返回查找对应的节点* @param head 待操作的链表* @param data 需要查找的节点数据* @return 查找到的节点*/
NODE *cllist_find(const NODE *head, DATA data)
{//使用const指针避免意外修改节点const NODE *p = head;//空链表if(!head)return NULL;//遍历列表(确保至少执行一次)do{if(memcmp(&(p->data), &data, sizeof(DATA)) == 0){return (NODE*)p;     //强转,避免类型不匹配}p = p->next;}while(p != head);return NULL;
}/*** 根据newdata修改old对应的节点数据* @param head 待操作的链表* @param old 待修改的原数据* @param newdata 待修改的新数据* @return 成功返回0,否则返回-1*/
int cllist_update(const NODE *head, DATA old, DATA newdata)
{NODE *p = cllist_find(head, old);if(!p){fprintf(stderr, "数据未找到!\n");return -1;}//找到旧数据的节点更新为新数据p->data = newdata;return 0;
}/*** 根据data删除对应的节点* @param head 待操作的链表* @param data 待删除的节点数据* @return 成功返回0,否则返回-1*/
int cllist_delete(NODE **head, DATA data)
{//尾随法(p为当前节点,q为前驱节点)NODE *p = *head, *q = NULL;//空链表if(!*head)return -1;//遍历节点while(p){//找到要删除数据的位置if(memcmp(&(p->data), &data, sizeof(DATA)) == 0){//若要删除的节点是头节点if(q == NULL)     //指针还没有进行尾随,证明是头节点{q = p->next;     //获取头节点的下一个节点//如果此头节点是唯一的头节点(p的next执行自身--头节点)if(p->next == *head){//链表置空*head = NULL;}else     //链表中除了要删除的头节点还有其他节点{//替换法(复制下一节点的数据并删除下一节点)p->data = q->data;     //用头节点的后续节点数据替换头节点的数据,删除后续节点p->next = q->next;free(q);}return 0;}//如果要删除的节点是非头节点q->next = p->next;     //前驱节点跳过当前节点free(p);     //释放当前节点return 0;}q = p;p = p->next;//因为是循环链表,所以要想办法终结循环链表if(p == *head)break;}return -1;
}/*** 销毁整个单向循环链表* @param head 待销毁的链表*/
void cllist_destroy(NODE **head)
{//空链表if(!*head)return;//在链表中找节点就用指针尾随法NODE *p = *head, *q = NULL;     //p指向当前节点,q用于保存待释放节点while(p){q = p;p = p->next;free(q);//解除循环链表if(p == *head)break;}*head = NULL;     //将头节点置空,不然会产生野指针
}
app.c
#include "clList.h"int main(int argc,char *argv[])
{NODE *head = NULL;//测试创建和插入cllist_create(&head, 111);     //创建头节点cllist_insert(&head, 222);cllist_insert(&head, 333);cllist_insert(&head, 444);cllist_showAll(head);     //111 444 333 222cllist_insertAthead(&head, 222);cllist_insertAthead(&head, 333);cllist_insertAthead(&head, 444);cllist_showAll(head);     //444 333 222 111 444 333 222//测试更新cllist_update(head, 333, 3333);cllist_showAll(head);     //444 3333 222 111 444 333 222//测试删除cllist_delete(&head, 444);cllist_showAll(head);     //3333 222 111 444 333 222//销毁链表cllist_destroy(&head);cllist_showAll(head);     //空链表,没有数据return 0;
}

三、优缺点总结

  • 优点

    1. 动态内存:无需预分配固定大小,适合数据量不确定的场景。

    2. 高效操作

      • 头插/头删:O(1)时间复杂度

      • 尾插:O(1)(维护尾指针时)或O(n)

      • 中间插入:O(1)(定位后)

    3. 循环特性:适合周期性访问场景(如:轮询调度、循环缓冲区)

    4. 内存效率:按需分配,无扩容浪费

  • 缺点:

    1. 随机访问 :必须遍历,时间复杂度O(n)

    2. 存储开销 :每个节点需额外存储指针

    3. 循环陷阱 :未正确处理终止条件会导致死循环

    4. 缓存不友好 :节点内存不连续,访问速度低于数组。

    5. 边界处理 :需特殊处理头尾节点的指针更新。

四、常见问题

1、数据结构的应用场景

所有编程领域都会涉及,例如

  • 链表:内存管理、任务队列

  • 树/图:数据库索引、路径规划

  • 哈希表:高速缓存、字典实现

2、面试题--单次遍历找到倒数第50个节点

解法:使用双指针技术

  1. 初始化指针fastslow指向头节点

  2. fast先移动50步

  3. 然后fastslow同步移动,当fast到达末尾时,slow即指向倒数第50个节点

//单链表找倒数第k节点
NODE *findLastKth(NODE *head, int k)
{NODE *fast = head, *slow = head;//先让快指针前进k步for(int i = 0; i < k; i++){//若快指针提前变空,说明链表长度不足kif(!fast)return NULL;     fast = fast->next;}while(fast != NULL){//循环链表终止条件fast = fast->next;slow = slow->next;}return slow;
}
//循环链表找倒数第k节点
NODE *findLastKth_Circular(NODE *head, int k)
{if(!head || k <= 0)return NULL;NODE *fast = head, *slow = head;//快指针先快走n步for(int i = 0; i < k; i++){if(fast == head)return NULL;fast = fast->next;}//同步移动直到fast回到起点while(fast != head){fast = fast->next;slow = slow->next;}return slow;
}

时间复杂度O(n),空间复杂度O(1)

练习题

【1】(单向链表)

  • 题目:建立一个包含若干整数的单向链表,比如: 1,2,3,4,51,2,3,4,5

    通过某些算法将其中各个节点逆转,比如: 5,4,3,2,15,4,3,2,1

  • 分析:(约瑟夫环、单向循环链表)主要分析如何将链表倒转

  • 代码:

    /#include <stdio.h>
    #include <stdlib.h>#define MAX_LENGHT 20// 链表节点结构体
    typedef struct ListNode 
    {int data;               // 节点数据域struct ListNode *next;  // 指向下一节点的指针
    } Node;//尾插法创建链表
    Node* createList(int arr[], int size) {if (size == 0) return NULL;Node *head = NULL, *tail = NULL;for (int i = 0; i < size; i++) {Node *newNode = (Node*)malloc(sizeof(Node));newNode->data = arr[i];newNode->next = NULL;if (head == NULL) {  // 链表为空时初始化头尾指针head = tail = newNode;} else {             // 尾插法维护尾指针tail->next = newNode;tail = newNode;}}return head;
    }//迭代反转法(三指针法)
    Node* reverseByIteration(Node *head) {Node *prev = NULL, *current = head, *next = NULL;while (current) {next = current->next;  // 保存下一节点指针current->next = prev;  // 反转指针方向prev = current;        // 前驱指针后移current = next;        // 当前指针后移}return prev;  // 新头节点是原链表尾节点
    }//头插法反转
    Node* reverseByHeadInsert(Node *head) {Node *newHead = NULL, *current = head;while (current) {Node *next = current->next;   // 保存下一节点current->next = newHead;      // 头插操作newHead = current;            // 更新新头节点current = next;               // 移动当前指针}return newHead;
    }//递归反转法
    Node* reverseByRecursion(Node *current, Node *prev) {if (!current) return prev;        // 递归终止条件Node *next = current->next;       // 保存下一节点current->next = prev;             // 反转指针方向return reverseByRecursion(next, current); // 递归调用
    }//打印链表
    void printList(Node *head) {Node *current = head;while (current) {printf("%d\t", current->data);current = current->next;}printf("\n");
    }//释放内存
    void freeList(Node *head) {Node *current = head;while (current) {Node *temp = current;current = current->next;free(temp);}
    }int main(int argc,char *argv[])
    {//动态分配数组int *arr = (int*)malloc(MAX_LENGHT * sizeof(int));int n;printf("Please enter the number of digits you want to reverse:>");scanf("%d", &n);printf("Please enter %d numbers:>\n", n);for(int i = 0; i < n; i++){scanf("%d", &arr[i]);}// 创建原始链表Node *original = createList(arr, n);printf("原链表: ");printList(original);// 迭代反转测试Node *reversed1 = reverseByIteration(original);printf("迭代反转: ");printList(reversed1);// 头插法反转测试Node *reversed2 = reverseByHeadInsert(reversed1);printf("头插法反转: ");printList(reversed2);// 递归反转测试Node *reversed3 = reverseByRecursion(reversed2, NULL);printf("递归反转: ");printList(reversed3);freeList(reversed3);return 0;
    }
    

【2】约瑟夫环问题

  • 题目:罗马人占领乔塔帕特后,犹太人与 Josephus 及他的朋友躲到一个洞中,族人决定宁愿死 也不要被敌人找到,于是决定了一个自杀方式,所有人排成一个圆圈,由第 1 个人开始报数, 每报数到第 3 人该人就必须自杀,然后再由下一个重新报数,直到所有人都自杀身亡为止。

    然而Josephus和他的朋友并不想死,Josephus要他的朋友先假装遵从,他将朋友与自己安排在两个 特殊的位置,于是逃过了这场死亡游戏。

现在假设有n个人形成一个单向循环链表,求最后剩余的两个节点。

  • 解析:主要是分析有头结点和没有头结点,对本题的影响。一般情况下链表带头结点是比较方便的,但是 对于本题,约瑟夫环问题,带上头结点的链表反而不利于操作,画图讲解。

  • 约瑟夫环问题(Josephus Problem)是一个精度的数学与数据结构问题,其核心在于通过特定规则在循环队列中淘汰元素(删除元素),最终找到幸存者的位置

    • 基本设定:

      1. n个人围成一个环形队列,编号通常为1到n(或0到n-1)

      2. 从第一个人开始报数,每次数到m(或动态变化的密码值)时,该人出列

      3. 从出列者的下一位重新开始报数,每次数到m(或动态变化的密码值)时,该人出列

      4. 目标:确定幸存者的初始编号

    • 使用循环链表模拟环形队列,每次遍历到第m个节点时删除该节点

  • 代码:

    //yuesefu.h#ifndef _YUESEFU_H
    #define _YUESEFU_H#include <stdio.h>
    #include <stdlib.h>
    #include <string.h>//创建节点结构体
    typedef struct node
    {int number;     //人员编号(1~n)struct node *next;     //指针域
    }NODE;//创建无头循环链表
    NODE *createCircle(int n);//约瑟夫环淘汰逻辑
    void joseph(NODE **head, int m);#endif //_YUESEFU_H
    
    //yuesefu.c#include "yuesefu.h"//创建无头循环链表
    NODE *createCircle(int n)
    {//定义头指针和前驱节点指针(初始化为空)NODE *head = NULL, *prev = NULL;//循环创建n个节点,人员编号从1开始递增for(int i = 1; i <= n; i++){//创建新节点NODE *new_node = (NODE*)malloc(sizeof(NODE));//检查节点是否创建成功if(!new_node)return NULL;//设置节点编号new_node->number = i;//如果链表为空则将新节点设为头节点if(!head){//头指针执行新节点head = new_node;//头指针指向自身,形成单个节点的循环结构head->next = head;}//非空链表(尾插法插入元素)else{//将新节点插入到prev节点之后prev->next = new_node;     //前驱节点连接新节点new_node->next = head;     //新节点指向头节点形成闭环}//更新前驱节点为当前新节点prev = new_node;}return head;
    }//约瑟夫环淘汰逻辑
    void joseph(NODE **head, int m)
    {//指针尾随法寻找待删除节点NODE *p = *head, *q = NULL;//当剩余节点数>2时循环while(p->next->next != p){//移动m-1次寻找待删除节点for(int i = 1; i < m; i++){q = p;p = p->next;}//删除当前节点q->next = p->next;     //跳过当前节点printf("处死:%d\n", p->number);//释放被淘汰节点的内存free(p);//从下一节点继续p = q->next;}//输出最后两个幸存者printf("幸存者1:%d\t幸存者2:%d\n", p->number, p->next->number);
    }
    //app.c#include "yuesefu.h"int main()
    {//经典的约瑟夫问题参数int n = 41, m = 3;NODE *p = createCircle(n);joseph(&p, m);//释放最后两个节点free(p->next);free(p);return 0;
    }

相关文章:

数据结构(三)---单向循环链表

单向循环链表&#xff08;Circular Linked List&#xff09; 一、基本概念 循环链表是一种特殊的链表&#xff0c;其末尾节点的后继指针指向头结点&#xff0c;形成一个闭环。 循环链表的操作与普通链表基本一致&#xff0c;但需注意循环特性的处理。 二、代码实现 clList…...

HCIP-H12-821 核心知识梳理 (3)

从EBGP邻居接受的路由发送给IBGP邻居的时候&#xff0c;下一跳不会自动修改。一个 Route - Policy 最多可配置 65535个节点 BFD 单跳使用UDP3784端口多跳使用UDP 4784端口。 防火墙 Local&#xff1a;代表防火墙自身&#xff0c;处理防火墙本地发起或接收的流量 。 优先级 100I…...

Vue接口平台学习七——接口调试页面请求体

一、实现效果图及简单梳理 请求体部分的左边&#xff0c;展示参数&#xff0c;分text和file类型。 右边部分一个el-upload的上传文件按钮&#xff0c;一个table列表展示&#xff0c;一个显示框&#xff0c;用于预览选择的文件&#xff0c;点击可大图展示。 二、页面内容实现 …...

STM32

GPIO 输入输出模式 GPIO 输出描述 GPIO_Mode_Out_OD 开漏输出模式&#xff1a; 1.对输入数据寄存器的读访问可得到I/O状态 HAL输出输出模式 GPIO输出描述GPIO_MODE_OUTPUT_PP推挽输出GPIO_MODE_OUTPUT_OD开漏输出GPIO输入GPIO_PULLUP上拉输入 寄存器 GPIOx->ODR/IDR …...

linux如何用关键字搜索日志

在 Linux 系统中搜索日志是日常运维的重要工作&#xff0c;以下是几种常用的关键字搜索日志方法&#xff1a; 1. 基础 grep 搜索 bash 复制 # 基本搜索&#xff08;区分大小写&#xff09; grep "keyword" /var/log/syslog# 忽略大小写搜索 grep -i "error&…...

381_C++_decrypt解密数据、encrypt加密数据,帧头和数据buffer分开

仿照.cpp中将帧头和数据分开处理的方式来修改.cpp中的加密: if (StreamCipher::self().needEncrypt()) {// 创建加密缓冲区static std::vector...

MongoDB常见语句

目录 1. 增删改 2. 评估查询运算符 3. 比较查询运算符 4. 逻辑运算符 5. 元素运算符 6. 数组查询运算符 7. 字段更新操作符 8. 数组更新操作符 10. 聚合管道 1. 增删改 增 db.getCollection("Y").insert({"age": 10,name: "ces5"});//增…...

Kotlin学习记录2

Android Studio中的注意事项 本文为个人学习记录&#xff0c;仅供参考&#xff0c;如有错误请指出。本文主要记录在Android Studio中开发时遇到的问题和回答。 Fragment有哪些特性&#xff1f; Fragment 是 Android 开发中的一个重要组件&#xff0c;具有以下特性&#xff1a…...

如何通过工具实现流程自动化

通过自动化工具&#xff0c;企业可以显著提高工作效率、降低人为错误、节省时间和成本。现代企业的运营中&#xff0c;流程管理是确保工作顺畅的关键&#xff0c;而人工处理繁琐的流程不仅容易出错&#xff0c;还会消耗大量的时间和人力资源。通过使用适合的自动化工具&#xf…...

组合数哭唧唧

前言&#xff1a;手写一个简单的组合数&#xff0c;但是由于长期没写&#xff0c;导致一些细节没处理好 题目链接 #include<bits/stdc.h> using namespace std; #define endl "\n"#define int long longconst int N (int)2e510; const int Mod (int)1e97;int…...

LINUX基石

Vim编辑器Linux系统常用命令管理Linux实例软件源Nginx服务配置多站点Cron定时任务在Linux系统上安装图形化界面升级Linux ECS实例内核设置Linux实例的预留内存Linux系统中TCP/UDP端口测试方法进入Linux/FreeBSD系统的单用户模式 Vim编辑器 linux系统默认安装vim编辑器。终端中…...

Flowable工程化改造相关文档

本章将针对前期进行的Flowable流程引擎研究&#xff0c;进行相应的工程化改造&#xff0c;改造过程分别为对Flowable引擎流程文件远程化处理&#xff0c;流程过程接口化升级&#xff0c;等方面进行改造&#xff0c;以适配其他项目对流程引擎的API调用 首先对流程引擎项目主要流…...

架构设计系列

架构设计系列&#xff1a;什么是架构设计架构设计系列&#xff1a;几个常用的架构设计原则架构设计系列&#xff1a;高并发系统的设计目标架构设计系列&#xff1a;如何设计可扩展架构架构设计系列&#xff1a;如何设计高性能架构架构设计系列&#xff1a;如何设计高可用架构架…...

波束形成(BF)从算法仿真到工程源码实现-第十节-非线性波束形成

一、概述 本节我们基于webrtc的非线性波束形成进行代码仿真&#xff0c;并对仿真结果进行展示和分析总结。更多资料和代码可以进入https://t.zsxq.com/qgmoN &#xff0c;同时欢迎大家提出宝贵的建议&#xff0c;以共同探讨学习。 二、仿真代码 2.1 常量参数 % *author : a…...

QuickAPI 全生命周期管理:从开发到退役的闭环实践​

数据 API 作为企业核心的数据资产&#xff0c;其生命周期管理直接影响数据服务的稳定性、安全性和复用效率。麦聪 QuickAPI 通过可视化、智能化的管理工具&#xff0c;构建了覆盖 API 全生命周期的闭环管理体系&#xff0c;实现从 "粗放式开发" 到 "精细化运营&…...

STM32 TDS+温度补偿

#define POLAR_CONSTANT (513385) /* 电导池常数&#xff0c;可通过与标准TDS测量仪对比计算反推 */ #define TDS_COEFFICIENT (55U) /* TDS 0.55 * 电子传导率*/void TDS_Value_Conversion() {u32 ad0;u8 i;float compensationCoefficient;float compens…...

【四川省第三届青少年C++算法设计大赛 (小低组) 第 一试】

一、单项选择题(共15题&#xff0c;每题2分&#xff0c;共计30分;每题有且仅有一个正确选项) 1、计算机中负责执行算术和逻辑运算的部件是() A. 内存 B.CPU C.硬盘 D.鼠标 2、近期备受关注的国产开源生成式人工智能大模型是() A. AlphaChat B. …...

疾控01-实验室信息管理系统需求分析

支持录入送检单位的基本信息&#xff0c;包括单位名称、联系方式、地址、联系人等。支持修改、删除、查询功能&#xff1b;支持录入检验目的的具体内容&#xff0c;如疾病类型&#xff08;例如血液检测、癌症检测&#xff09;或样本来源&#xff08;如水质监测、食品安全检测&a…...

Redis之RedLock算法以及底层原理

自研redis分布式锁存在的问题以及面试切入点 lock加锁关键逻辑 unlock解锁的关键逻辑 使用Redis的分布式锁 之前手写的redis分布式锁有什么缺点&#xff1f;&#xff1f; Redis之父的RedLock算法 Redis也提供了Redlock算法&#xff0c;用来实现基于多个实例的分布式锁。…...

【JavaScript】二十二、通过关系查找DOM节点、新增、删除

文章目录 1、DOM节点的分类2、查找亲戚节点2.1 父节点查找2.2 子节点查找2.3 兄弟节点查找 3、新增节点3.1 创建新节点3.2 追加节点3.3 克隆节点3.4 案例&#xff1a;学成在线页面数据渲染 4、删除节点 1、DOM节点的分类 DOM树里每一个内容都称之为节点&#xff0c;节点分为三…...

SQL学习-关联查询(应用于多表查询)

复习 前几篇写的基础查询语法复习 以上都在单一表单内进行查询&#xff0c;那么我们需要用到多个表单的数据时&#xff0c;我们应该怎么处理呢&#xff1f; 关联查询 在excle文档中我们的处理方式如下 excle的这个查询虽然简单直观&#xff0c;但是也具有一定的局限性 比…...

在 MySQL 单表存储 500 万数据的场景下,如何设计读取

在 MySQL 单表存储 500 万数据的场景下,设计高效读取方案需要从 查询优化、架构扩展、硬件调优 三个层面综合考虑。以下是具体方案,结合实际项目经验(如标易行投标服务平台)进行分析: 一、查询优化:降低单次查询开销 1. 索引优化 核心原则:仅为高频查询条件、排序字段、…...

Python使用FastMCP开发MCP服务端

MCP简介 Model Context Protocol (MCP) 是一个专门为 LLM&#xff08;大语言模型&#xff09;应用设计的协议&#xff0c;它允许你构建服务器以安全、标准化的方式向 LLM 应用程序公开数据和功能。FastMCP 作为 Python 生态中的一款轻量级框架&#xff0c;利用装饰器来简化路由…...

ESLint常见错误

1、Strings must use singlequote —— 字符串必须使用单引号 2、Extra semicolon semi——额外的分号&#xff1a;一行语句结尾不能添加分号 3、Unexpected trailing comma —— 行尾多了一个逗号 4、Newline required at end of file but not found ——文件结尾必须要新加…...

京东硬核挑战潜规则,外卖算法要变天?

刘强东这次回归后的动作&#xff0c;真是越来越有看头了&#xff01;最近那段内部讲话视频爆出来&#xff0c;直接扔了个重磅炸弹&#xff1a;京东外卖&#xff0c;净利润率永远不许超过5%&#xff0c;谁敢超标就得挨处分&#xff01;这话一出&#xff0c;整个外卖圈估计都得抖…...

怎样利用 macOS 自带功能快速进行批量重命名文件教程

在日常办公或个人使用中&#xff0c;我们经常需要对多个文件进行重命名操作。幸运的是&#xff0c;macOS 提供了一套非常实用的内置工具&#xff0c;可以轻松完成这一任务而无需借助任何第三方应用程序。今天&#xff0c;我们就来详细介绍如何利用 macOS 自带的功能实现文件的批…...

Java Spring Cloud框架使用及常见问题

Spring Cloud作为基于Spring Boot的分布式微服务框架&#xff0c;显著简化了微服务架构的开发与管理。其核心优势包括集成Eureka、Ribbon、Hystrix等组件&#xff0c;提供一站式服务发现、负载均衡、熔断容错等解决方案&#xff0c;支持动态配置与消息总线&#xff0c;实现高效…...

机器视觉检测Pin针歪斜应用

在现代电子制造业中&#xff0c;Pin针&#xff08;插针&#xff09;是连接器、芯片插座、PCB板等元器件的关键部件。如果Pin针歪斜&#xff0c;可能导致接触不良、短路&#xff0c;甚至整机失效。传统的人工检测不仅效率低&#xff0c;还容易疲劳漏检。 MasterAlign 机器视觉对…...

抗量子算法验证工具

抗量子算法计算工具 抗量子算法验证工具ML-KEMML-DSASLH-DSA 抗量子算法验证工具 2024年末&#xff0c;美国NIST陆续公布了FIPS-203、FIPS-204、FIPS-205算法标准文档&#xff0c;抽空学习了一下&#xff0c;做了个算法计算工具。 ML-KEM ML-DSA SLH-DSA 需要的朋友可留言交流…...

临床协调简历模板

模板信息 简历范文名称&#xff1a;临床协调简历模板&#xff0c;所属行业&#xff1a;其他 | 职位&#xff0c;模板编号&#xff1a;C1S3WO 专业的个人简历模板&#xff0c;逻辑清晰&#xff0c;排版简洁美观&#xff0c;让你的个人简历显得更专业&#xff0c;找到好工作。希…...

linux命令八

tmux防止远程管理中断 格式:tmux # 进入会话模式 进入会话模式后,你进行文件的压缩时,如果远程管理突然中断,也不会影响压缩的进程 DNS服务器 作用&#xff1a;负责域名解析的服务器&#xff0c;将域名解析为IP地址 /etc/resolv.conf:指定DNS服务器地址配置文件 日志管理 •常见…...

37-串联所有单词的子串

给定一个字符串 s 和一个字符串数组 words。 words 中所有字符串 长度相同。 s 中的 串联子串 是指一个包含 words 中所有字符串以任意顺序排列连接起来的子串。 例如&#xff0c;如果 words ["ab","cd","ef"]&#xff0c; 那么 "abcdef…...

机器学习赋能的多尺度材料模拟与催化设计前沿技术

随着新能源、先进制造等领域对功能材料性能要求的日益严苛&#xff0c;传统材料研发模式面临显著挑战&#xff1a;跨尺度关联机制不清晰、实验试错周期长、计算资源消耗巨大。人工智能技术与多尺度模拟方法的深度融合&#xff0c;为材料科学开辟了“数据驱动物理建模”的创新路…...

HarmonyOS-ArkUI V2工具类:AppStorageV2:应用全局UI状态存储

AppStorageV2是一个能够跨界面存储数据,管理数据的类。开发者可以使用AppStorageV2来存储全局UI状态变量数据。它提供的是应用级的全局共享能力,开发者可以通过connect绑定同一个key,进行跨ability数据共享。 概述 AppStorageV2是一个单例,创建时间是应用UI启动时。其目的…...

【Linux】进程池bug、命名管道、systemV共享内存

一.进程池bug 我们在之前进程池的创建中是通过循环创建管道&#xff0c;并且让子进程与父进程关闭不要的读写段以构成通信信道。但是我们这样构建的话会存在一个很深的bug。 我们在销毁进程池时是先将所有的信道的写端关闭&#xff0c;让其子进程read返回值为0&#xff0c;并…...

.Net 9 webapi使用Docker部署到Linux

参考文章连接&#xff1a; https://www.cnblogs.com/kong-ming/p/16278109.html .Net 6.0 WebApi 使用Docker部署到Linux系统CentOS 7 - 长白山 - 博客园 项目需要跨平台部署&#xff0c;所以就研究了一下菜鸟如何入门Net跨平台部署&#xff0c;演示使用的是Net 9 webAPi Li…...

【差分隐私相关概念】瑞丽差分隐私(RDP)引理1

引理1的详细推导过程 引理1陈述 若分布 P P P 和 Q Q Q 满足&#xff1a; D ∞ ( P ∥ Q ) ≤ ϵ 且 D ∞ ( Q ∥ P ) ≤ ϵ , D_\infty(P \parallel Q) \leq \epsilon \quad \text{且} \quad D_\infty(Q \parallel P) \leq \epsilon, D∞​(P∥Q)≤ϵ且D∞​(Q∥P)≤ϵ, …...

Java练习——day1(反射)

文章目录 练习1练习2练习3思考封装原则与反射合理使用反射“破坏”封装的场景 练习1 编写代码&#xff0c;通过反射获取String类的所有公共方法名称&#xff0c;并按字母顺序打印。 示例代码&#xff1a; import java.lang.reflect.Method; import java.util.Arrays;public …...

【C++】二叉搜索树

目录 一、二叉搜索树 &#x1f354;二叉搜索树概念 &#x1f35f;二叉搜索树的操作 &#x1f32e;二叉搜索树的实现 &#x1f96a;二叉搜索树的应用 &#x1f959;二叉搜索树的效率分析 二、结语 一、二叉搜索树 &#x1f354;二叉搜索树概念 二叉搜索树又称二叉排序树&…...

fastjson2 使用bug

fastjson2 版本2.0.52 转jsonString保留null值求助 有如下对象&#xff1a; JSONObject jsonObject {“A”:null,“B”:“value”} 当服务运行几天之后&#xff0c; 还是这个json格式&#xff0c;因为需要保留null值&#xff0c;如下方法&#xff1a; jsonObject.toJSONString…...

Redis日常维护技巧与常见问题解决方案

Redis是一个开源的内存数据存储系统&#xff0c;广泛应用于缓存、消息队列、实时分析等场景。由于其高性能和持久化特性&#xff0c;越来越多的企业开始引入Redis。然而&#xff0c;要使Redis高效、稳定地运行&#xff0c;日常的维护和问题解决显得尤其重要。本文将分享一些Red…...

【Leetcode-Hot100】最小覆盖子串

题目 解答 想到使用双指针哈希表来实现&#xff0c;双指针的left和right控制实现可满足字符串。 class Solution(object):def minWindow(self, s, t):""":type s: str:type t: str:rtype: str"""len_s, len_t len(s), len(t)hash_map {}for…...

【Sequelize】关联模型和孤儿记录

一、关联模型的核心机制 1. 关联类型与组合规则 • 基础四类型&#xff1a; • hasOne&#xff1a;外键存储于目标模型&#xff08;如用户档案表存储用户ID&#xff09; • belongsTo&#xff1a;外键存储于源模型&#xff08;如订单表存储用户ID&#xff09; • hasMany&…...

系统分析师-第三遍-章节导图

导图要求&#xff1a; 第一章 绪论 第二章 数学与工程基础 导图要不偏瘫...

算法(ALGORITHMS)---- 关于阶乘

Everyday life is different,even with your state and mind!So if i have some new ways or logic to make a good Algorithms,I gonna post it and share with U guys! If there is anything error aboubt what I demonstrated,pls speak out on the comment,Thanks! 一.最初…...

电路(b站石群老师主讲,持续更新中...)

文章目录 第一章 电路模型和电路定律1.1电路和电路模型 第一章 电路模型和电路定律 第一章的重点&#xff1a; 1.电压、电流的参考方向 2.电阻元件和电源元件的特性 3.基尔霍夫定律&#xff08;KCL,KVL,&#xff09; KCL&#xff1a;基尔霍夫电流定律 KVL&#xff1a;基尔…...

Python multiprocessing模块Pool类介绍

multiprocessing.Pool 类是 Python 中用于并行处理任务的强大工具,它可以创建一个进程池,允许你在多个进程中并行执行任务,从而充分利用多核 CPU 的性能。下面为你总结 Pool 类的常用方法。 1. 创建进程池 from multiprocessing import Pool pool = Pool(processes=None)参…...

CCF CSP 第36次(2024.12)(1_移动_C++)

CCF CSP 第36次&#xff08;2024.12&#xff09;&#xff08;1_移动_C&#xff09; 解题思路&#xff1a;思路一&#xff1a; 代码实现代码实现&#xff08;思路一&#xff09;&#xff1a; 时间限制&#xff1a; 1.0 秒 空间限制&#xff1a; 512 MiB 原题链接 解题思路&…...

【教程】PyTorch多机多卡分布式训练的参数说明 | 附通用启动脚本

转载请注明出处&#xff1a;小锋学长生活大爆炸[xfxuezhagn.cn] 如果本文帮助到了你&#xff0c;欢迎[点赞、收藏、关注]哦~ 目录 torchrun 一、什么是 torchrun 二、torchrun 的核心参数讲解 三、torchrun 会自动设置的环境变量 四、torchrun 启动过程举例 机器 A&#…...

九、自动化函数02

// 进阶版本的屏幕截图 void getScreenShot(String str) throws IOException {// ./src/test/image/ 存放图片路径// 屏幕截图SimpleDateFormat sim1 new SimpleDateFormat("yyyy-MM-dd");SimpleDateFormat sim2 new SimpleDateFormat("HHmmss");Str…...