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

数据结构全解析:从线性到非线性,优缺点与应用场景深度剖析

1. 线性数据结构

(1)数组(Array)(适合静态数据)

优点

  1. 随机访问高效:通过索引可以直接访问元素,时间复杂度为 O(1)。

  2. 内存连续:数组在内存中是连续存储的,缓存友好,访问速度快。

  3. 简单易用:实现简单,适合存储固定大小的数据。

缺点

  1. 固定大小:数组的大小通常是固定的,动态调整需要重新分配内存,效率较低。

  2. 插入和删除效率低:在数组中插入或删除元素需要移动其他元素,时间复杂度为 O(n)。

  3. 内存浪费:如果数组未完全使用,会造成内存浪费。

代码示例:

#include <iostream>
using namespace std;int main() {int arr[5] = {1, 2, 3, 4, 5}; // 静态数组cout << "Element at index 2: " << arr[2] << endl; // 随机访问// 插入元素(低效)for (int i = 4; i >= 2; i--) {arr[i + 1] = arr[i];}arr[2] = 10; // 在索引2处插入10// 打印数组for (int i = 0; i < 5; i++) {cout << arr[i] << " ";}return 0;
}

(2)链表(Linked List)(适合动态数据)

优点

  1. 动态大小:链表可以动态分配内存,无需预先指定大小。

  2. 插入和删除高效:在已知节点的情况下,插入和删除操作的时间复杂度为 O(1)。

  3. 内存利用率高:链表按需分配内存,不会造成内存浪费。

缺点

  1. 随机访问效率低:链表不支持随机访问,查找元素需要遍历,时间复杂度为 O(n)。

  2. 额外空间开销:链表需要额外的指针空间来存储节点之间的关系。

  3. 缓存不友好:链表节点在内存中不连续,访问速度较慢。

代码示例:

#include <iostream>
using namespace std;struct Node {int data;Node* next;
};void printList(Node* head) {while (head != nullptr) {cout << head->data << " ";head = head->next;}cout << endl;
}int main() {Node* head = new Node{1, nullptr};head->next = new Node{2, nullptr};head->next->next = new Node{3, nullptr};// 插入元素Node* newNode = new Node{10, head->next};head->next = newNode;printList(head); // 输出: 1 10 2 3return 0;
}

(3)栈(Stack)

优点

  1. 简单高效:栈只允许在栈顶进行操作,实现简单,插入和删除的时间复杂度为 O(1)。

  2. 适合特定问题:栈非常适合解决递归问题、括号匹配、表达式求值等问题。

缺点

  1. 功能受限:栈只能在一端进行操作,无法直接访问中间元素。

  2. 容量限制:栈的容量通常是固定的(如果使用数组实现),动态调整需要额外开销。

代码示例:

#include <iostream>
#include <stack>
using namespace std;int main() {stack<int> s;s.push(10); // 入栈s.push(20);s.push(30);cout << "Top element: " << s.top() << endl; // 访问栈顶s.pop(); // 出栈cout << "Stack size: " << s.size() << endl; // 栈大小return 0;
}

(4)队列(Queue)

优点

  1. 先进先出(FIFO):队列适合处理需要按顺序处理的数据,如任务调度、缓冲区等。

  2. 高效操作:在队尾插入和队头删除的时间复杂度为 O(1)。

缺点

  1. 功能受限:队列只能在一端插入,另一端删除,无法直接访问中间元素。

  2. 容量限制:队列的容量通常是固定的(如果使用数组实现),动态调整需要额外开销。

代码示例:

#include <iostream>
#include <queue>
using namespace std;int main() {queue<int> q;q.push(10); // 入队q.push(20);q.push(30);cout << "Front element: " << q.front() << endl; // 访问队头q.pop(); // 出队cout << "Queue size: " << q.size() << endl; // 队列大小return 0;
}

(5)双端队列(Deque)

优点

  1. 灵活操作:双端队列允许在两端进行插入和删除操作,功能更强大。

  2. 高效操作:在两端插入和删除的时间复杂度为 O(1)。

缺点

  1. 实现复杂:相比栈和队列,双端队列的实现更复杂。

  2. 容量限制:如果使用数组实现,容量通常是固定的,动态调整需要额外开销。

代码示例:

#include <iostream>
#include <deque>
using namespace std;int main() {deque<int> dq;// 在队头和队尾插入元素dq.push_front(10); // 队头插入dq.push_back(20);  // 队尾插入dq.push_back(30);cout << "Front element: " << dq.front() << endl; // 访问队头cout << "Back element: " << dq.back() << endl;   // 访问队尾// 删除队头和队尾元素dq.pop_front(); // 删除队头dq.pop_back();  // 删除队尾cout << "Size of deque: " << dq.size() << endl; // 输出: 1return 0;
}

2. 非线性数据结构

(1)树(Tree)

优点

  1. 层次结构:树适合表示具有层次关系的数据,如文件系统、组织结构等。

  2. 高效搜索:二叉搜索树(BST)的搜索、插入和删除操作的时间复杂度为 O(log n)。

  3. 动态调整:树可以动态调整结构,适合频繁插入和删除的场景。

缺点

  1. 复杂度高:树的实现和操作比线性结构复杂。

  2. 平衡问题:如果树不平衡(如退化为链表),性能会下降,搜索时间复杂度退化为 O(n)。

  3. 额外空间开销:树需要额外的指针空间来存储节点之间的关系。

代码示例:

#include <iostream>
using namespace std;struct TreeNode {int data;TreeNode* left;TreeNode* right;TreeNode(int val) : data(val), left(nullptr), right(nullptr) {}
};// 插入节点
TreeNode* insert(TreeNode* root, int val) {if (root == nullptr) {return new TreeNode(val);}if (val < root->data) {root->left = insert(root->left, val);} else {root->right = insert(root->right, val);}return root;
}// 中序遍历(左-根-右)
void inorderTraversal(TreeNode* root) {if (root == nullptr) return;inorderTraversal(root->left);cout << root->data << " ";inorderTraversal(root->right);
}int main() {TreeNode* root = nullptr;root = insert(root, 10);root = insert(root, 5);root = insert(root, 15);root = insert(root, 3);root = insert(root, 7);cout << "Inorder Traversal: ";inorderTraversal(root); // 输出: 3 5 7 10 15return 0;
}

(2)二叉树(Binary Tree)

优点

  1. 简单高效:二叉树的结构简单,适合实现二叉搜索树、堆等数据结构。

  2. 搜索效率高:在平衡的情况下,搜索、插入和删除的时间复杂度为 O(log n)。

缺点

  1. 平衡问题:如果二叉树不平衡,性能会下降。

  2. 空间开销:需要额外的指针空间来存储左右子节点。

高级数据结构  约等于 基于二叉树的数据结构

代码示例:

#include <iostream>
using namespace std;struct TreeNode {int data;TreeNode* left;TreeNode* right;
};void inorderTraversal(TreeNode* root) {if (root == nullptr) return;inorderTraversal(root->left);cout << root->data << " ";inorderTraversal(root->right);
}int main() {TreeNode* root = new TreeNode{10, nullptr, nullptr};root->left = new TreeNode{5, nullptr, nullptr};root->right = new TreeNode{15, nullptr, nullptr};cout << "Inorder Traversal: ";inorderTraversal(root); // 输出: 5 10 15return 0;
}

(3)图(Graph)

优点

  1. 表示复杂关系:图适合表示多对多的关系,如社交网络、地图等。

  2. 灵活性强:图可以是有向的或无向的,可以带权或不带权。

缺点

  1. 实现复杂:图的实现和操作比线性结构和树更复杂。

  2. 空间开销大:图的存储(如邻接矩阵或邻接表)需要较大的空间。

  3. 算法复杂度高:图的遍历和搜索算法(如 DFS、BFS)的时间复杂度较高。

代码示例:

#include <iostream>
#include <vector>
using namespace std;class Graph {int V; // 顶点数vector<vector<int>> adj; // 邻接表public:Graph(int V) : V(V), adj(V) {}void addEdge(int u, int v) {adj[u].push_back(v);adj[v].push_back(u); // 无向图}void printGraph() {for (int i = 0; i < V; i++) {cout << "Vertex " << i << ": ";for (int j : adj[i]) {cout << j << " ";}cout << endl;}}
};int main() {Graph g(4);g.addEdge(0, 1);g.addEdge(0, 2);g.addEdge(1, 3);g.printGraph();return 0;
}

(4)堆(Heap)

优点

  1. 高效获取极值:堆可以在 O(1) 时间内获取最大值或最小值。

  2. 适合优先级队列:堆适合实现优先级队列,用于任务调度等场景。

  3. 动态调整:堆可以动态调整结构,插入和删除的时间复杂度为 O(log n)。

缺点

  1. 功能受限:堆只能快速获取极值,无法高效访问其他元素。

  2. 实现复杂:堆的实现比线性结构复杂。

代码示例:

#include <iostream>
#include <vector>
#include <algorithm>
using namespace std;class MaxHeap {vector<int> heap;void heapifyUp(int index) {while (index > 0) {int parent = (index - 1) / 2;if (heap[parent] < heap[index]) {swap(heap[parent], heap[index]);index = parent;} else {break;}}}void heapifyDown(int index) {int size = heap.size();while (true) {int left = 2 * index + 1;int right = 2 * index + 2;int largest = index;if (left < size && heap[left] > heap[largest]) {largest = left;}if (right < size && heap[right] > heap[largest]) {largest = right;}if (largest != index) {swap(heap[index], heap[largest]);index = largest;} else {break;}}}public:void push(int val) {heap.push_back(val);heapifyUp(heap.size() - 1);}void pop() {if (heap.empty()) return;heap[0] = heap.back();heap.pop_back();heapifyDown(0);}int top() {if (heap.empty()) throw runtime_error("Heap is empty");return heap[0];}bool empty() {return heap.empty();}
};int main() {MaxHeap heap;heap.push(10);heap.push(30);heap.push(20);heap.push(5);cout << "Max element: " << heap.top() << endl; // 输出: 30heap.pop();cout << "Max element after pop: " << heap.top() << endl; // 输出: 20return 0;
}

(5)哈希表(Hash Table)(散列表)

优点

  1. 高效查找:哈希表的查找、插入和删除操作的平均时间复杂度为 O(1)。

  2. 适合大规模数据:哈希表适合存储和查找大规模数据。

缺点

  1. 哈希冲突:哈希冲突(不同的数据,计算出来的哈希值可能相同,从而导致冲突)会影响性能,需要额外的处理(如链地址法或开放地址法)。

  2. 空间开销大:哈希表需要较大的空间来减少冲突。

  3. 无序性:哈希表中的元素是无序的,无法直接按顺序访问。

雪崩效应:

  • 如果两个数据有一点不同,它们的哈希值就会差别很大,从而不容易冲突。
  • 哈希表的空间效率和时间效率是矛盾的,使用的空间越大,越容易设计哈希函数。
  • 如果空间很小,再好的哈希函数也会产生冲突。
  • 在使用哈希表时,需要在空间和时间效率上取得平衡。

代码示例:

#include <iostream>
#include <unordered_map>
using namespace std;int main() {unordered_map<string, int> hashMap;hashMap["Alice"] = 25;hashMap["Bob"] = 30;cout << "Age of Alice: " << hashMap["Alice"] << endl; // 输出: 25cout << "Age of Bob: " << hashMap["Bob"] << endl; // 输出: 30return 0;
}

总结

  • 线性数据结构:适合处理顺序关系明确的数据,操作简单高效,但功能受限。

  • 非线性数据结构:适合处理复杂关系的数据,功能强大,但实现和操作复杂度较高。

相关文章:

数据结构全解析:从线性到非线性,优缺点与应用场景深度剖析

1. 线性数据结构 &#xff08;1&#xff09;数组&#xff08;Array&#xff09;&#xff08;适合静态数据&#xff09; 优点&#xff1a; 随机访问高效&#xff1a;通过索引可以直接访问元素&#xff0c;时间复杂度为 O(1)。 内存连续&#xff1a;数组在内存中是连续存储的&…...

《使用 Python Flask + MySQL + ECharts 构建销售数据看板》实战案例笔记

《使用 Python Flask + MySQL + ECharts 构建销售数据看板》实战案例笔记 技术栈说明 后端:Python 3.10 + Flask 框架数据库:MySQL前端:ECharts 5.4 + HTML/CSS数据可视化:柱状图 / 折线图 / 饼图 / 雷达图项目结构 project/ ├── server.py # 后端服务 └──…...

StringBuilder和StringJoiner的运用

package test12; import java.util.Scanner; import java.util.StringJoiner;public class Test { public static void main(String[] args) {/* String str "你玩的真好&#xff0c;下次别玩了&#xff0c;TMD&#xff0c;CNM";String[] arr {"TMD", &…...

科技创新:改变生活的力量与未来趋势

人工智能在智能客服中的应用越来越普遍。它改变了传统的客服模式。AI可以快速回答用户的问题&#xff0c;提高了客服效率和服务质量。 首先&#xff0c;人工智能能够处理大量信息。智能客服可以在几秒钟内回应客户的请求。这比人工客服快得多。客户不需要等待很久就能得到答案…...

Maven指定JDK

在使用 Maven 管理 Java 项目时&#xff0c;有时需要指定使用特定的 JDK 版本。这通常是因为项目需要与特定版本的 JDK 兼容&#xff0c;或者在不同的开发环境中需要确保使用正确的 JDK 版本。通常来说在IDEA工具中设置了正确的JDK版本&#xff0c;使用IDEA编译也不会有任何异常…...

Jenkins持续集成与Web前端、SpringBoot项目的部署

Jenkins是一个开源的持续集成&#xff08;Continuous Integration, CI&#xff09;和持续交付&#xff08;Continuous Delivery, CD&#xff09;工具&#xff0c;广泛应用于软件开发过程中。‌它基于Java开发&#xff0c;旨在提供一个开放易用的软件平台&#xff0c;帮助软件项…...

如何使用Opentelemetry+jaeger对Go与Java项目实现分布式链路追踪

本文介绍![如何使用Opentelemetryjaeger实现分布式链路追踪] 关于opentelemetry的介绍可以看下面的文章 https://blog.csdn.net/qq_62368250/article/details/143516314本文中相关图片以及源代码地址 https://github.com/wuchenyanghaoshuai/others/blob/main/step39/README.…...

LabVIEW闭环控制系统硬件选型与实时性能

在LabVIEW闭环控制系统的开发中&#xff0c;硬件选型直接影响系统的实时性、精度与稳定性。需综合考虑数据采集速度&#xff08;采样率、接口带宽&#xff09;、计算延迟&#xff08;算法复杂度、处理器性能&#xff09;、输出响应时间&#xff08;执行器延迟、控制周期&#x…...

Html5学习教程,从入门到精通, HTML5超链接应用的详细语法知识点和案例代码(18)

HTML5超链接应用的详细语法知识点和案例代码 超链接&#xff08;Hyperlink&#xff09;&#xff0c;也称为跃点链接&#xff0c;是互联网和文档编辑中的一种重要概念。 超链接的定义 超链接是指从一个网页指向一个目标的连接关系&#xff0c;这个目标可以是另一个网页&#…...

STM32 HAL库 CAN过滤器配置

之前在STM32 f407 CAN收发 基于HAL库和Cubemx配置_stm32f407can收发程序-CSDN博客这篇博文里写了一下配置CAN收发的方法&#xff0c;当时由于并没有使用过滤器的现实需求&#xff0c;所以就也没仔细研究。现在工作中确实需要用到过滤器了&#xff0c;有些项目中控制器和发动机E…...

【前端面试题】Vu3常见的面试题

1.Vue3与 Vue2的核心区别有哪些&#xff1f; ‌ 响应式系统 ‌&#xff1a; ‌ Vue2&#xff1a;通过Object.defineProperty 实现响应式。这种方式在处理对象属性的添加和删除时存在局限性&#xff0c;且无法直接监控数组的变化 ‌;‌Vue3&#xff1a;采用Proxy 实现响应式&…...

【数据分享】2001-2024年我国逐年植被净初级生产力(NPP)数据

植被净初级生产力&#xff08;Net Primary Productivity&#xff0c;NPP&#xff09;是生态学中的一个重要概念&#xff0c;表示单位面积植被在特定时间内吸收的净光合有机物&#xff0c;是衡量生态系统中植物通过光合作用所产生的有机物质减去植物呼吸作用消耗的有机物质的量&…...

java-正则表达式

一、正则表达式能干什么&#xff1f; ✅ 验证格式&#xff1a;手机号、邮箱、日期✅ 提取数据&#xff1a;从日志/文本中抓取关键信息✅ 替换文本&#xff1a;批量修改字符串内容 二、Java正则核心API Java中用 java.util.regex 包的两个类&#xff1a; Pattern&#xff1a;…...

正则表达式(复习)

文章目录 一、[]: 一个字符集合二、{}: 重复次数三、特殊符号四、(): 分组五、python代码示例六、注意 正则表达式(regular expression)描述了一种字符串匹配的模式&#xff08;pattern&#xff09;&#xff0c;可以用来检查一个串是否含有某种子串、将匹配的子串替换或者从某个…...

sonarqube+SonarScanner+postpresql+jenkins

本地执行参考这篇 sonarqubeSonarScannerPostgreSQL实现代码质量扫描&#xff08;windows&#xff09;_sonarqube 10.7部署-CSDN博客 集成到Jenkins中操作如下 在 Jenkins 中安装 SonarQube Scanner 和配置 Job 1、安装 SonarQube Scanner 插件&#xff1a; 在 Jenkins 的管理…...

牛客周赛Round 84

本场比赛难度不大&#xff0c;也是本人第一次AK&#xff0c;最后一题用组合数学推公式&#xff0c;本篇博客主要讲解最后一题的思路 G-小红的陡峭值&#xff08;五&#xff09;&#xff08;hard&#xff09;_牛客周赛 Round 84 思路&#xff1a;本题要我们求p/q mod M ,即p*&am…...

TDengine 配置 ODBC 数据源

简介 TDengine ODBC 是为 TDengine 实现的 ODBC 驱动程序&#xff0c;支持 Windows 系统的应用&#xff08;如 PowerBI 等&#xff09;以及用户自定义开发的应用程序&#xff0c;通过 ODBC 标准接口访问本地、远程和云服务的 TDengine 数据库。 TDengine ODBC 提供基于 WebSo…...

numpy常用函数详解

在深度神经网络代码中经常用到numpy库的一些函数&#xff0c;很多看过之后很容易忘记&#xff0c;本文对经常使用的函数进行归纳总结。 np.arange arange是numpy一个常用的函数&#xff0c;该函数主要用于创建等差数列。它的使用方法如下所示&#xff1a; numpy.arange([star…...

deepseek 3FS编译

3FS在ubuntu22.04下的编译&#xff08;记录下编译过程&#xff0c;方便后续使用&#xff09; 环境信息 OS ubuntu 22.04内核版本 6.8.0-52-genericlibfuse 3.16.1rust 1.75.0FoundationDB 7.1.66meson 1.0.0ninja 1.10.1 libfuse编译 以下建议均在root下执行 pip3 install…...

【CXX】6.2 str — rust::Str

Rust::Str 公共 API // rust/cxx.hclass Str final { public:Str() noexcept;Str(const Str &) noexcept;Str(const String &) noexcept;// 如果输入不是 UTF-8&#xff0c;抛出 std::invalid_argument 异常。Str(const std::string &);Str(const char *);Str(con…...

从555到正弦波:手把手教你用立创EDA仿真+打样一个2KHz波形发生器(附完整工程)

从555到正弦波&#xff1a;立创EDA全流程打造2KHz波形发生器实战指南 在电子设计领域&#xff0c;波形发生器是最基础却又最考验设计功底的经典项目之一。想象一下&#xff0c;当你亲手设计的电路板输出完美的正弦波时&#xff0c;那种成就感绝非购买现成模块可比。本文将带你用…...

3个实用技巧:让Mermaid图表创作效率翻倍的秘密武器

3个实用技巧&#xff1a;让Mermaid图表创作效率翻倍的秘密武器 【免费下载链接】mermaid mermaid-js/mermaid: 是一个用于生成图表和流程图的 Markdown 渲染器&#xff0c;支持多种图表类型和丰富的样式。适合对 Markdown、图表和流程图以及想要使用 Markdown 绘制图表和流程图…...

HDMI设备开发必看:EDID/E-EDID数据结构全解析(附实战代码)

HDMI设备开发实战&#xff1a;EDID/E-EDID二进制解析与工程实现 当你的HDMI设备无法正确识别显示器分辨率时&#xff0c;屏幕闪烁或黑屏的瞬间是否让你抓狂&#xff1f;作为连接数字世界的桥梁&#xff0c;EDID&#xff08;Extended Display Identification Data&#xff09;就…...

告别DWA!用TEB局部规划器让你的ROS机器人学会‘倒车入库’(附多机编队避障实测对比)

告别DWA&#xff01;用TEB局部规划器解锁机器人高阶机动能力 在机器人自主导航领域&#xff0c;传统动态窗口方法(DWA)长期占据主导地位&#xff0c;直到开发者们遇到那些需要倒车、急转弯或狭窄空间多机协作的真实场景。想象一下仓储机器人需要在货架间完成"倒车入库&quo…...

Qwen2.5-VL-7B-Instruct应用场景:跨境电商商品图自动打标+多语种描述生成

Qwen2.5-VL-7B-Instruct应用场景&#xff1a;跨境电商商品图自动打标多语种描述生成 1. 跨境电商的痛点与解决方案 跨境电商卖家每天面临两个核心挑战&#xff1a;商品图片标注和多语言描述撰写。传统方式需要人工逐张图片添加标签&#xff0c;再用翻译工具转换语言&#xff…...

wan2.1-vae提示词评估体系:构建BLEU-Style指标量化中文提示词有效性

wan2.1-vae提示词评估体系&#xff1a;构建BLEU-Style指标量化中文提示词有效性 1. 为什么需要评估提示词质量 在AI图像生成领域&#xff0c;提示词的质量直接影响最终生成效果。好的提示词能准确表达创作意图&#xff0c;而模糊或不当的提示词可能导致生成结果与预期不符。特…...

Iceoryx(冰羚):无锁队列与并发控制的设计与实现3(源码解析)

接上篇设计4: 索引管理层&#xff08; MpmcIndexQueue / CyclicIndex&#xff09;Subscriber存储数据使用的是queue&#xff0c;是为了保证数据的读取顺序。MpmcLockFreeQueue 为了满足多个进程同时写的情况&#xff0c;采用了索引数据分离的方案&#xff08;底层的索引实现为 …...

OpenClaw对话增强:nanobot镜像的聊天历史持久化方案

OpenClaw对话增强&#xff1a;nanobot镜像的聊天历史持久化方案 1. 为什么需要对话持久化 作为一个长期使用OpenClaw进行自动化任务的开发者&#xff0c;我经常遇到这样的困扰&#xff1a;当需要执行一个跨越数小时甚至数天的长周期任务时&#xff0c;传统的短对话模式会导致…...

SpringCloud Alibaba与Nacos版本不匹配?手把手教你解决‘Client not connected‘错误

SpringCloud Alibaba与Nacos版本兼容性实战&#xff1a;彻底解决Client not connected问题 微服务架构的复杂性往往隐藏在细节之中。当SpringCloud Alibaba项目启动时控制台突然抛出Client not connected, current status:STARTING的红色警告&#xff0c;不少开发者都会心头一紧…...

从CLPM到RI-CLPM:Mplus中交叉滞后模型的进阶指南与选择策略

从CLPM到RI-CLPM&#xff1a;纵向数据分析的模型选择与实战解析 在心理学和行为科学的纵向研究中&#xff0c;交叉滞后模型&#xff08;CLPM&#xff09;长期以来是分析变量间相互影响关系的标准工具。然而&#xff0c;随着研究方法论的进步&#xff0c;研究者们逐渐认识到传统…...