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

算法题——合并 k 个升序的链表

题目描述:
 

合并 k 个升序的链表并将结果作为一个升序的链表返回其头节点。

数据范围:节点总数 0≤n≤50000≤n≤5000,每个节点的val满足 ∣val∣<=1000∣val∣<=1000

要求:时间复杂度 O(nlogn)


一、常见解法

(一)暴力排序法

暴力排序法的思路非常直接。首先,将链表数组中的所有链表进行遍历,把它们头尾相接组成一个长链表。这个过程可以通过遍历链表数组,对于每个链表,依次将其节点添加到新的长链表中。然后,对这个长链表进行排序。这种方法虽然简单易懂,但是时间复杂度比较高。因为在组成长链表的过程中,需要遍历整个链表数组,时间复杂度为 O(k * n),其中 是链表数组的长度, n是平均每个链表的长度。而对长链表进行排序的时间复杂度取决于排序算法的选择,例如快速排序的平均时间复杂度为 O(nlogn),其中 n是长链表的长度,也就是所有链表长度之和。所以总体时间复杂度较高,但对于一些小规模的问题或者对时间要求不高的场景,这种方法仍然可以尝试。

(二)暴力求解优化法

暴力求解优化法是对暴力排序法的一种改进。它的核心思想是每次取出 K个链表中最小的元素,不断放入结果数组中。具体实现时,可以通过遍历 K个链表的首元素,找到其中最小的元素,将其加入结果数组,然后将该最小元素所在链表的指针向后移动一位。重复这个过程,直到所有链表都为空。这种方法在一定程度上减少了不必要的排序操作,但是时间复杂度仍然较高。每次查找最小元素需要遍历 K个链表,时间复杂度为 O(k),假设总共有 n个元素,那么总体时间复杂度为 O(nk)。

(三)分治法

分治法是一种更加高效的方法。将 k个链表配对并将同一对中的链表合并。第一轮合并以后, k个链表被合并成了 k/2个链表,平均长度为 2n/k,然后是 k/4个链表,k/8 个链表等等。重复这一过程,直到我们得到了最终的有序链表。具体实现时,可以使用递归的方式进行合并。首先将链表数组分成两部分,分别对这两部分进行合并,然后再将合并后的结果进行合并。这种方法的时间复杂度为 O(n log k),其中 n是所有链表的总长度,k 是链表数组的长度。因为每次合并的时间复杂度为 O(n),而总共需要进行 log k次合并。

(四)堆排序或构建二叉排序树法

堆排序或构建二叉排序树法为解决这个问题提供了不同的思路。对于堆排序,可以使用小根堆来存储链表的首元素。每次从堆中取出最小的元素,将其加入结果链表,然后将该元素所在链表的下一个元素加入堆中。这样可以保证每次取出的元素都是当前未处理元素中的最小值。时间复杂度为 O(n log k),其中 n是所有链表的总长度,k 是链表数组的长度。对于构建二叉排序树,可以将每个链表的元素依次插入二叉排序树中,然后进行中序遍历得到有序链表。这种方法的时间复杂度取决于二叉排序树的构建和遍历,一般情况下时间复杂度也为 O(n log k)。具体的实现可以参考相关博客中的更基础概念和示例代码。

二、代码实现

(一)暴力排序法

#include <stdio.h>
#include <stdlib.h>// 链表节点结构
struct ListNode {int val;struct ListNode* next;
};// 创建新节点
struct ListNode* createNode(int val) {// 分配内存空间给新节点struct ListNode* newNode = (struct ListNode*)malloc(sizeof(struct ListNode));// 设置新节点的值newNode->val = val;// 新节点的下一个指针初始化为 NULLnewNode->next = NULL;return newNode;
}// 插入节点到链表尾部
void insertNode(struct ListNode** head, int val) {// 创建新节点struct ListNode* new = createNode(val);// 如果链表为空,将新节点设为链表的头节点if (*head == NULL) {*head = new;return;}// 临时指针指向链表的头节点struct ListNode* temp = *head;// 遍历链表找到最后一个节点while (temp->next!= NULL) {temp = temp->next;}// 将新节点连接到链表的末尾temp->next = new;
}// 合并 k 个链表(暴力排序法)
struct ListNode* mergeKListsBruteForce(struct ListNode** lists, int listsSize) {// 初始化结果链表为空struct ListNode* result = NULL;// 遍历输入的链表数组for (int i = 0; i < listsSize; i++) {// 获取当前链表struct ListNode* currentList = lists[i];// 遍历当前链表的所有节点while (currentList!= NULL) {// 将当前链表的节点值插入到结果链表中insertNode(&result, currentList->val);// 移动到当前链表的下一个节点currentList = currentList->next;}}// 对合并后的链表进行排序struct ListNode* current = result;while (current!= NULL) {// 获取当前节点的下一个节点struct ListNode* next = current->next;while (next!= NULL) {// 如果当前节点的值大于下一个节点的值,则交换它们的值if (current->val > next->val) {int temp = current->val;current->val = next->val;next->val = temp;}// 移动到下一个节点next = next->next;}// 移动到下一个节点current = current->next;}return result;
}

(二)暴力求解优化法

#include <stdio.h>
#include <stdlib.h>// 链表节点结构
struct ListNode {int val;struct ListNode* next;
};// 创建新节点
struct ListNode* createNode(int val) {// 分配内存空间给新节点struct ListNode* newNode = (struct ListNode*)malloc(sizeof(struct ListNode));// 设置新节点的值newNode->val = val;// 新节点的下一个指针初始化为 NULLnewNode->next = NULL;return newNode;
}// 插入节点到链表尾部
void insertNode(struct ListNode** head, int val) {// 创建新节点struct ListNode* new = createNode(val);// 如果链表为空,将新节点设为链表的头节点if (*head == NULL) {*head = new;return;}// 临时指针指向链表的头节点struct ListNode* temp = *head;// 遍历链表找到最后一个节点while (temp->next!= NULL) {temp = temp->next;}// 将新节点连接到链表的末尾temp->next = new;
}// 查找 k 个链表中的最小节点所在的链表索引
int findMinIndex(struct ListNode** lists, int listsSize) {// 初始化最小索引为 -1,表示还未找到最小节点所在的链表int minIndex = -1;// 初始化最小值为整数最大值int minVal = __INT_MAX__;// 遍历所有链表for (int i = 0; i < listsSize; i++) {// 如果当前链表不为空且当前链表的头节点值小于当前最小值if (lists[i]!= NULL && lists[i]->val < minVal) {// 更新最小值minVal = lists[i]->val;// 更新最小索引minIndex = i;}}return minIndex;
}// 合并 k 个链表(暴力求解优化法)
struct ListNode* mergeKListsOptimized(struct ListNode** lists, int listsSize) {// 创建一个虚拟头节点struct ListNode* dummy = createNode(0);// 当前节点指针,初始指向虚拟头节点struct ListNode* current = dummy;// 记录非空链表的数量int nonEmptyLists = listsSize;// 当还有非空链表时循环while (nonEmptyLists > 0) {// 找到 k 个链表中最小节点所在的链表索引int minIndex = findMinIndex(lists, listsSize);// 如果找到了非空链表if (minIndex!= -1) {// 将最小节点连接到结果链表中current->next = lists[minIndex];// 当前节点指针后移current = current->next;// 移动最小链表的头指针到下一个节点lists[minIndex] = lists[minIndex]->next;// 如果该链表为空,减少非空链表数量if (lists[minIndex] == NULL) {nonEmptyLists--;}}}// 返回合并后的链表(不包括虚拟头节点)return dummy->next;
}

(三)分治法

#include <stdio.h>
#include <stdlib.h>// 链表节点结构
struct ListNode {int val;struct ListNode* next;
};// 创建新节点
struct ListNode* createNode(int val) {// 分配内存空间给新节点struct ListNode* newNode = (struct ListNode*)malloc(sizeof(struct ListNode));// 设置新节点的值newNode->val = val;// 新节点的下一个指针初始化为 NULLnewNode->next = NULL;return newNode;
}// 插入节点到链表尾部
void insertNode(struct ListNode** head, int val) {// 创建新节点struct ListNode* new = createNode(val);// 如果链表为空,将新节点设为链表的头节点if (*head == NULL) {*head = new;return;}// 临时指针指向链表的头节点struct ListNode* temp = *head;// 遍历链表找到最后一个节点while (temp->next!= NULL) {temp = temp->next;}// 将新节点连接到链表的末尾temp->next = new;
}// 合并两个升序链表
struct ListNode* mergeTwoLists(struct ListNode* l1, struct ListNode* l2) {// 创建一个虚拟头节点struct ListNode dummy;// 指向虚拟头节点的指针,用于构建合并后的链表struct ListNode* tail = &dummy;// 当两个链表都不为空时循环while (l1 && l2) {// 如果第一个链表当前节点值小于第二个链表当前节点值if (l1->val < l2->val) {// 将第一个链表当前节点连接到结果链表tail->next = l1;// 移动第一个链表指针到下一个节点l1 = l1->next;} else {// 将第二个链表当前节点连接到结果链表tail->next = l2;// 移动第二个链表指针到下一个节点l2 = l2->next;}// 移动结果链表指针到新连接的节点tail = tail->next;}// 将剩余的非空链表连接到结果链表末尾tail->next = l1? l1 : l2;// 返回合并后的链表(不包括虚拟头节点)return dummy.next;
}// 分治法合并 k 个链表
struct ListNode* mergeKListsDivideAndConquer(struct ListNode** lists, int start, int end) {// 如果起始索引等于结束索引,说明只有一个链表,直接返回该链表if (start == end) {return lists[start];}// 如果起始索引大于结束索引,说明没有链表可合并,返回 NULLif (start > end) {return NULL;}// 计算中间索引int mid = start + (end - start) / 2;// 递归合并左半部分链表struct ListNode* left = mergeKListsDivideAndConquer(lists, start, mid);// 递归合并右半部分链表struct ListNode* right = mergeKListsDivideAndConquer(lists, mid + 1, end);// 合并左右两部分已合并的链表return mergeTwoLists(left, right);
}

(四)堆排序或构建二叉排序树法

#include <stdio.h>
#include <stdlib.h>// 链表节点结构
struct ListNode {int val;struct ListNode* next;
};// 创建新节点
struct ListNode* createNode(int val) {// 分配内存空间给新节点struct ListNode* newNode = (struct ListNode*)malloc(sizeof(struct ListNode));// 设置新节点的值newNode->val = val;// 新节点的下一个指针初始化为 NULLnewNode->next = NULL;return newNode;
}// 插入节点到链表尾部
void insertNode(struct ListNode** head, int val) {// 创建新节点struct ListNode* new = createNode(val);// 如果链表为空,将新节点设为链表的头节点if (*head == NULL) {*head = new;return;}// 临时指针指向链表的头节点struct ListNode* temp = *head;// 遍历链表找到最后一个节点while (temp->next!= NULL) {temp = temp->next;}// 将新节点连接到链表的末尾temp->next = new;
}// 交换两个整数的值
void swap(int* a, int* b) {// 临时变量存储其中一个整数的值int temp = *a;// 将另一个整数的值赋给第一个整数*a = *b;// 将临时变量的值赋给第二个整数*b = temp;
}// 堆调整函数
void heapify(int arr[], int n, int i) {// 初始化当前最大元素的索引为 iint largest = i;// 左子节点的索引int left = 2 * i + 1;// 右子节点的索引int right = 2 * i + 2;// 如果左子节点存在且比当前最大元素大if (left < n && arr[left] > arr[largest])largest = left;// 如果右子节点存在且比当前最大元素大if (right < n && arr[right] > arr[largest])largest = right;// 如果最大元素不是当前节点if (largest!= i) {// 交换当前节点和最大元素节点的值swap(&arr[i], &arr[largest]);// 递归调整新的子树heapify(arr, n, largest);}
}// 构建堆
void buildHeap(int arr[], int n) {// 从最后一个非叶子节点开始调整堆for (int i = n / 2 - 1; i >= 0; i--)heapify(arr, n, i);
}// 合并 k 个链表(堆排序法)
struct ListNode* mergeKListsHeap(struct ListNode** lists, int listsSize) {// 创建最小堆数组int minHeap[listsSize];int heapSize = 0;// 将每个链表的第一个节点的值加入最小堆数组for (int i = 0; i < listsSize; i++) {if (lists[i]!= NULL) {minHeap[heapSize++] = lists[i]->val;}}// 构建最小堆buildHeap(minHeap, heapSize);// 创建虚拟头节点struct ListNode dummy;struct ListNode* tail = &dummy;// 当堆不为空时循环while (heapSize > 0) {// 获取堆顶元素(最小元素)int minVal = minHeap[0];// 将最小元素加入结果链表tail->next = createNode(minVal);tail = tail->next;// 在链表数组中找到值为最小元素的链表,并将其头节点后移for (int i = 0; i < listsSize; i++) {if (lists[i]!= NULL && lists[i]->val == minVal) {lists[i] = lists[i]->next;if (lists[i]!= NULL) {// 如果该链表不为空,将新的头节点值加入堆minHeap[0] = lists[i]->val;} else {// 如果该链表为空,用堆的最后一个元素替换堆顶元素,并减小堆大小minHeap[0] = minHeap[--heapSize];}// 调整堆heapify(minHeap, heapSize, 0);break;}}}// 返回合并后的链表(不包括虚拟头节点)return dummy.next;
}

三、总结展望

1.暴力排序法

  • 优点:思路简单直接,容易理解和实现。对于小规模问题或者对时间要求不高的场景,可以快速解决问题。
  • 缺点:时间复杂度高,特别是在处理大规模数据或者链表数量较多的情况时,效率低下。

2.暴力求解优化法

  • 优点:相比暴力排序法,减少了不必要的排序操作,在一定程度上提高了效率。对于一些特定的输入数据,可能会有较好的表现。
  • 缺点:时间复杂度仍然较高,每次查找最小元素需要遍历所有链表,当链表数量较多时,效率会受到很大影响。

3.分治法

  • 优点:时间复杂度为 ,相比暴力方法有很大的提升。通过递归地合并链表,充分利用了链表已经升序排列的特点,减少了比较次数。
  • 缺点:实现相对复杂,需要理解递归的思想和过程。在处理小规模问题时,可能由于递归调用的开销,效率不一定比其他方法高。

4.堆排序或构建二叉排序树法

  • 优点:时间复杂度为 ,能够快速找到当前最小的元素,保证合并后的链表始终是升序的。对于大规模数据和链表数量较多的情况,具有较好的性能。
  • 缺点:需要构建和维护堆或二叉排序树,增加了空间复杂度和实现的复杂性。对于一些简单的场景,可能过于复杂。

(二)未来探索方向

  1. 进一步优化时间复杂度:可以探索更高效的算法和数据结构,以降低合并 k 个升序链表的时间复杂度。例如,研究新的分治策略或者改进堆排序的实现,以减少比较次数和操作次数。
  2. 空间复杂度的优化:在保证时间效率的同时,尽量降低空间复杂度。可以考虑使用更紧凑的数据结构或者优化算法的内存使用,以减少内存占用。
  3. 结合实际应用场景进行优化:根据不同的应用场景,对算法进行针对性的优化。例如,在数据库管理系统中,可以考虑利用数据库的索引结构来加速链表的合并;在图形处理中,可以根据图形的特点和需求,选择合适的合并策略。
  4. 并行计算的应用:对于大规模数据和链表数量较多的情况,可以考虑使用并行计算技术,将合并操作分配到多个处理器上同时进行,以提高效率。
  5. 动态调整算法:根据输入数据的特点和规模,动态地选择最合适的合并算法。例如,当链表数量较少时,可以使用简单的暴力方法;当链表数量较多时,自动切换到分治或堆排序方法。

        总之,合并 k 个升序链表是一个具有挑战性的问题,通过不断地探索和创新,可以找到更高效、更实用的解决方案,为实际应用提供更好的支持。

相关文章:

算法题——合并 k 个升序的链表

题目描述&#xff1a; 合并 k 个升序的链表并将结果作为一个升序的链表返回其头节点。 数据范围&#xff1a;节点总数 0≤n≤50000≤n≤5000&#xff0c;每个节点的val满足 ∣val∣<1000∣val∣<1000 要求&#xff1a;时间复杂度 O(nlogn) 一、常见解法 &#xff08…...

智能制造与精益制造的模型搭建

现行制造模式分析I-痛点改善思路-管控省优四化推行...

快速生成生产级Go应用的利器——Cgapp

简介 CGAPP是一个强大的命令行工具&#xff0c;开发者通过简单的命令就可以快速搭建起一个完整的Go项目框架。这个框架不仅包括后端服务&#xff0c;还可以集成前端代码和数据库配置&#xff0c;大大简化了项目的初始化过程。 安装 安装CGAPP的过程非常简单。首先&#xff0…...

MySQL基本语法、高级语法知识总结以及常用语法案例

MySQL基本语法总结 MySQL是一种广泛使用的关系型数据库管理系统&#xff0c;其基本语法涵盖了数据库和数据表的创建、查询、修改和删除等操作。 一、数据库操作 创建数据库&#xff08;CREATE DATABASE&#xff09; 语法&#xff1a;CREATE DATABASE [IF NOT EXISTS] databa…...

单片机(学习)2024.10.11

目录 按键 按键原理 按键消抖 1.延时消抖 2.抬手检测 通信 1.通信是什么 2.电平信号和差分信号 3.通信的分类 (1)时钟信号划分 同步通信 异步通信 (2)通信方式划分 串行通信 并行通信 (3)通信方向划分 单工 半双工 全双工 4.USART和UART&#xff08;串口通信&a…...

Java创建型模式(二)——工厂模式(简单工厂模式、工厂方法模式、抽象工厂模式、工厂模式扩展等完整详解,附有代码——案例)

文章目录 五.工厂模式5.1 概述5.2简单工厂模式5.2.1 概述5.2.2 结构5.2.3 实现5.2.4 优缺点5.2.5 扩展—静态工厂 5.3 工厂方法模式5.3.1概述5.3.2 结构5.3.3 实现5.3.4 优缺点 5.4 抽象工厂模式5.4.1 概述5.4.2 结构5.4.3 实现5.4.4 优缺点5.4.5 使用场景 5.5 工厂模式扩展 五…...

C++学习,容器类 <set>

C 标准库中的 <set> 是一个关联容器&#xff0c;它存储了一组唯一的元素&#xff0c;并按照一定的顺序进行排序。<set> 提供了高效的元素查找、插入和删除操作。它是基于红黑树实现的&#xff0c;因此具有对数时间复杂度的查找、插入和删除性能。 声明集合&#x…...

Cisco Catalyst 9000 交换产品系列 IOS XE 17.15.1 发布下载,新增功能概览

Cisco Catalyst 9000 Series Switches, IOS XE Release 17.15.1 ED 思科 Catalyst 9000 交换产品系列 IOS XE 系统软件 请访问原文链接&#xff1a;https://sysin.org/blog/cisco-catalyst-9000/&#xff0c;查看最新版。原创作品&#xff0c;转载请保留出处。 作者主页&…...

Python知识点:基于Python技术,如何使用MMDetection进行目标检测

开篇&#xff0c;先说一个好消息&#xff0c;截止到2025年1月1日前&#xff0c;翻到文末找到我&#xff0c;赠送定制版的开题报告和任务书&#xff0c;先到先得&#xff01;过期不候&#xff01; 使用MMDetection进行目标检测的Python技术详解 MMDetection是一个开源的目标检测…...

Chromium HTML Tags与c++接口对应关系分析

一、HTML 标签(HTML Tags) <a> <head> <img>等等这些标签在c中的接口是如何定义和查找的呢&#xff1f; 更多标签参考&#xff1a; HTML <a> target 属性 (w3school.com.cn) 二、html_tag_names.json5 (third_party\blink\renderer\core\html\htm…...

React Fiber 解析:前端性能提升密码

文章目录 背景React 采用 fiber 主要为了解决哪些问题&#xff1f;性能问题&#xff1a;用户体验问题&#xff1a; 为什么在 React 15 版本中性能会差&#xff1a;浏览器绘制原理&#xff1a;react 15 架构和问题 那么 fiber 怎么解决了这个问题&#xff1f;任务“大”的问题递…...

【吊打面试官系列-微服务面试题】微服务架构如何运作?

大家好&#xff0c;我是锋哥。今天分享关于【微服务架构如何运作&#xff1f;】面试题&#xff0c;希望对大家有帮助&#xff1b; 微服务架构如何运作&#xff1f; 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 微服务架构是一种将单一应用程序构建为一组小型、独…...

Windows系统编程 - 目录操作、磁盘、卷信息

前言 各位师傅大家好&#xff0c;我是qmx_07&#xff0c;今天继续讲解Windows系统编程的相关知识:目录操作 目录 目录操作 创建目录 通过CreateDirectory函数创建目录 原型: BOOL CreateDirectoryA([in] LPCSTR lpPathName,//目录名称[in, opt…...

搭建SaaS知识库:优化教育机构的在线教学效能

随着信息技术的飞速发展&#xff0c;教育领域正经历着前所未有的变革。在线教学已成为教育机构提供灵活学习体验、扩大覆盖范围、提升教学效率的重要手段。然而&#xff0c;如何在海量资源与复杂教学场景中高效管理知识&#xff0c;确保教学质量&#xff0c;成为教育机构面临的…...

CSS中backdrop-filter详解

文章目录 CSS中backdrop-filter详解一、引言二、backdrop-filter基础1、基本概念1.1、基本语法 2、滤镜函数2.1、代码示例 三、实际应用1、创建模糊背景1.1、代码示例 2、结合其他CSS属性2.1、代码示例 四、总结 CSS中backdrop-filter详解 一、引言 在现代网页设计中&#xf…...

AI测试入门:理解 LLM 的基准测试(Benchmark)

AI测试入门:理解 LLM 的基准测试(Benchmark) 1. 基准测试的定义2. 基准测试的目的3. 基准测试的常用指标4. 基准测试的流程5. 常用的AI基准测试框架总结1. 基准测试的定义 LLM 的基准测试是一种评估 LLM 的标准化方法,通过使用预定义的数据集、任务和评估指标,对LLM 在特定…...

InternVid:用于多模态视频理解与生成的大规模视频-文本数据集 | ICLR Spotlight

InternVid 是一个开源的大规模视频-文本数据集&#xff0c;旨在促进视频理解和生成任务的发展&#xff0c;由上海人工智能实验室与南京大学、中国科学院等单位联合发布&#xff0c;相关的工作已经被ICLR2024接收。它包含超过 700 万个视频&#xff0c;总时长近 76 万小时&#…...

Hive数仓操作(十)

一、Hive 分页查询 在大数据处理中&#xff0c;分页查询是非常常见的需求。Hive 提供了 LIMIT 和 OFFSET 关键字来方便地进行分页操作。本文将详细介绍它们的用法。 1. 基本用法 LIMIT&#xff1a;用于限制查询结果的行数。OFFSET&#xff1a;用于指定从哪一行开始检索。 2…...

Android 扩大View的点击区域

文章目录 Android 扩大View的点击区域使用padding属性使用TouchDelegate使用getLocationOnScreen监听 Android 扩大View的点击区域 使用padding属性 通过设置 padding 属性扩大点击区域。 使用&#xff1a; <?xml version"1.0" encoding"utf-8"?&…...

[Qt学习笔记] 解决QTextEdit数据过多UI卡死问题

背景问题 在项目中使用QTextEdit显示软件的日志信息&#xff0c;由于在连续输出日志信息&#xff0c;刚开始QTextEdit显示没什么问题&#xff0c;长时间就会出现UI界面卡死&#xff0c;内存占用变高。晚上查了说QTextEdit的append函数如果不释放会累计增加内存&#xff0c;包括…...

OgreNext高级材质中增加线宽,点大小,虚线模式绘制支持

修改Ogre高级材质系统&#xff0c;增加线宽&#xff0c;点大小&#xff0c;虚线模式&#xff0c;虚线参数的支持,效果如下&#xff1a; 需要修改的代码文件如下&#xff1a; 修改如下 代码文本&#xff1a; //范围[0.2 - 51] 0.2 * [0,255];Ogre::uint8 mLineWidth;//范围[…...

STM32中的DMA数据转运——下篇

STM32中的DMA数据转运——上篇-CSDN博客 在上篇文章中&#xff0c;我们讨论了STM32中的DMA&#xff08;直接存储器访问&#xff09;及其工作原理、存储器类型和总线设计。接下来&#xff0c;我们将更深入地探讨DMA的具体配置方法、常见应用场景以及一些实际设计中的注意事项。…...

51单片机的智能小区安防系统【proteus仿真+程序+报告+原理图+演示视频】

1、主要功能 该系统由AT89C51/STC89C52单片机LCD1602显示模块时钟模块温度传感器烟雾传感器CO传感器红外感应传感器IC卡蓝牙继电器按键、蜂鸣器、LED等模块构成。适用于智能小区安防、智能家居安防等相似项目。 可实现功能: 1、LCD1602实时显示北京时间、温度、烟雾浓度和CO浓…...

数仓建模流程

数仓建模简介 一句话总结 数仓建模中的“建模”是一个将数据有序组织和存储起来的过程&#xff0c;旨在提高数据的使用效率和降低使用成本。 详细描述 在数仓建模中&#xff0c;“建模”指的是构建数据模型&#xff0c;也就是数据的组织和存储方法。数据模型强调从业务、数…...

Neo4j CQL语句 使用教程

CREATE命令 : CREATE (<node-name>:<label-name>{ <Property1-name>:<Property1-Value>........<Propertyn-name>:<Propertyn-Value>} )字段说明 CREATE (dept:Dept { deptno:10,dname:“Accounting”,location:“Hyderabad” })&#…...

STM32-HAL库 驱动DS18B20温度传感器 -- 2024.10.8

目录 一、教程简介 二、驱动理论讲解 三、CubeMX生成底层代码 四、Keil5编写代码 五、实验结果 一、教程简介 本教程面向初学者&#xff0c;只介绍DS18B20的常用功能&#xff0c;但也能满足大部分的运用需求。跟着本教程操作&#xff0c;可在10分钟内解决DS18b20通信难题。…...

HTML 符号

HTML 符号 HTML(超文本标记语言)是一种用于创建网页的标准标记语言。它使用一系列的标签来描述网页的结构和内容。HTML 符号,通常指的是 HTML 标签,是构成 HTML 文档的基础。本文将详细介绍 HTML 符号的概念、种类、用途以及如何在网页设计中正确使用它们。 HTML 符号的概…...

编译后的MySQL安装

MySQL安装 1.下载网址2.下载方式3.配置配置环境变量修改配置文件初始化安装服务启动服务测试修改 Mysql 默认密码 1.下载网址 https://dev.mysql.com/downloads/mysql/2.下载方式 选择对应版本下载 mysql-xxx-winx64.zip&#xff0c;该压缩包为编译后文件&#xff0c;并非源码…...

Ubuntu安装Apache教程

系统版本&#xff1a;Ubuntu版本 23.04 Ubuntu是一款功能强大且用户友好的操作系统&#xff0c;而Apache是一款广泛使用的Web服务器软件。在Ubuntu上安装Apache可以帮助用户搭建自己的网站或者进行Web开发。为大家介绍如何在Ubuntu上安装Apache&#xff0c;并提供详细的教程和操…...

Nginx跳转模块之location与rewrite

目录 一、location模块与rewrite模块区别 二、location模块的基本介绍 1. location模块是什么&#xff1f; 2. 三种匹配类别 3. 常用的匹配规则 4. 匹配优先级 三、location模块使用实例 1.精准匹配优先级小于一般匹配的特殊情况 2 .解决方法 3. 实际网站使用中的三个匹配…...