数据结构与算法-分治算法

数据结构与算法

数据结构与算法是计算机科学中的两个核心概念,它们在软件开发和问题解决中起着至关重要的作用。

数据结构

数据结构是计算机中存储、组织和管理数据的方式,它能够帮助我们高效地访问和修改数据。不同的数据结构适用于不同类型的应用场景。

常见的数据结构包括:

  • 数组:一种线性数据结构,用于存储具有相同类型的元素集合,每个元素在内存中占据连续的位置。
  • 链表:由节点组成的线性数据结构,每个节点包含数据和指向下一个节点的指针。
  • 栈:一种后进先出(LIFO)的数据结构,常用于管理函数调用、表达式求值等。
  • 队列:一种先进先出(FIFO)的数据结构,适用于任务调度、缓冲处理等场景。
  • 树:一种分层数据结构,由节点组成,每个节点可以有零个或多个子节点。
  • 图:由顶点(节点)和边组成,可以表示多对多的关系,适用于网络分析、路径查找等。

算法

算法是解决特定问题的一系列步骤和规则。算法的性能通常通过时间复杂度和空间复杂度来衡量。算法的设计和选择对程序的效率有很大影响。

常见的算法类型包括:

  • 排序算法:如快速排序、归并排序、堆排序等,用于将数据集合按特定顺序排列。
  • 搜索算法:如二分搜索、深度优先搜索(DFS)、广度优先搜索(BFS)等,用于在数据结构中查找特定元素。
  • 图算法:如Dijkstra算法、A*搜索算法、Prim算法和Kruskal算法等,用于解决图中的最短路径、最小生成树等问题。
  • 动态规划:一种通过将问题分解为重叠的子问题来解决问题的方法,适用于具有最优子结构特性的问题。
  • 分治算法:将问题分解为若干个规模较小的子问题,递归解决子问题后合并结果,适用于某些特定类型的优化问题。

分治算法

分治算法是一种处理复杂问题的方法,它的核心思想是将一个大问题分解成若干个规模较小但类似于原问题的子问题,递归或迭代地解决这些子问题,然后将子问题的解合并以得到原问题的解。这种方法的关键在于“分”和“治”两个步骤:分是将问题分解,治是独立解决分解后的小问题。
分治算法通常适用于解决那些可以通过重复应用相同解决方案的问题,例如数组排序、矩阵乘法、快速幂计算等。

  • 分治算法示意图
+-------------------+     +-------------------+     +-------------------+
|                   |     |                   |     |                   |
|   原问题(n个元素)  |     |   子问题1(n/2个元素) |     |   子问题2(n/2个元素)  |
|                   |     |                   |     |                   |
+-------------------+     +-------------------+     +-------------------+
           |                            |                            |
           v                            v                            v
+-------------------+     +-------------------+     +-------------------+
|                   |     |                   |     |                   |
|   子问题1继续分解  |     |   子问题1继续分解  |     |   子问题2继续分解  |
|  (n/4个元素)      |     |  (n/4个元素)      |     |  (n/4个元素)      |
|                   |     |                   |     |                   |
+-------------------+     +-------------------+     +-------------------+
           |                            |                            |
           v                            v                            v
+-------------------+     +-------------------+     +-------------------+
|                   |     |                   |     |                   |
|   最小子问题(1个元素) |     |   最小子问题(1个元素) |     |   最小子问题(1个元素) |
|                   |     |                   |     |                   |
+-------------------+     +-------------------+     +-------------------+
           |                            |                            |
           +------------+------------+            +------------+------------+
                          合并操作                        合并操作                        合并操作

分治算法特点

  • 分解:将原问题分解为若干个规模较小的相同类型的问题。
  • 独立解决:递归或迭代地独立解决每个子问题。
  • 合并:将子问题的解合并,形成原问题的解。

应用实例

  • 归并排序(Merge Sort):一种分稳算法,将无序数组分为两半,递归地对两半进行排序,然后将排序好的两半合并成一个有序数组。
  • 快速排序(Quick Sort):通过选定一个基准元素,将数组分为两部分,一部分包含所有小于基准的元素,另一部分包含所有大于基准的元素,然后递归地对这两部分进行快速排序。
  • 快速幂算法(Fast Exponentiation):用于计算一个数的幂,通过将幂指数分解为若干个更小的幂指数,然后逐个计算并合并结果。
  • 二分查找(Binary Search):在有序数组中查找特定元素,通过将数组分为两半并比较中间元素与目标值,逐步缩小搜索范围。
  • Strassen矩阵乘法:一种矩阵乘法算法,通过分解和合并操作,减少了计算量,提高了计算效率。

注意事项

  • 分治算法可能不适用于所有问题,需要确保问题可以被分解为独立的子问题,并且子问题的解可以容易地合并。
  • 分治算法可能会引入额外的开销,如递归调用和数据复制,因此在某些情况下可能不是最优选择。
  • 需要仔细设计分解和合并策略,以确保算法的效率和正确性。

分治算法c++示例

  1. 归并排序:归并排序的时间复杂度为 O(n log n),其中 n 是数组的大小。这种算法的优势在于其稳定性,即相等的元素在排序后保持原来的相对顺序。
#include <iostream>
#include <vector>

// 合并两个有序数组
void merge(std::vector<int>& arr, int left, int mid, int right) {
    int n1 = mid - left + 1; // 左半部分的大小
    int n2 = right - mid;    // 右半部分的大小

    // 创建临时数组
    std::vector<int> L(n1), R(n2);

    // 拷贝数据到临时数组
    for (int i = 0; i < n1; i++)
        L[i] = arr[left + i];
    for (int j = 0; j < n2; j++)
        R[j] = arr[mid + 1 + j];

    // 合并临时数组回原数组
    int i = 0, j = 0, k = left;
    while (i < n1 && j < n2) {
        if (L[i] <= R[j]) {
            arr[k] = L[i];
            i++;
        } else {
            arr[k] = R[j];
            j++;
        }
        k++;
    }

    // 拷贝剩余的左半部分元素
    while (i < n1) {
        arr[k] = L[i];
        i++;
        k++;
    }

    // 拷贝剩余的右半部分元素
    while (j < n2) {
        arr[k] = R[j];
        j++;
        k++;
    }
}

// 归并排序的递归函数
void mergeSort(std::vector<int>& arr, int left, int right) {
    if (left < right) {
        // 找到中间位置
        int mid = left + (right - left) / 2;

        // 递归地对左右两部分进行排序
        mergeSort(arr, left, mid);
        mergeSort(arr, mid + 1, right);

        // 合并已排序的左右两部分
        merge(arr, left, mid, right);
    }
}

// 打印数组的函数
void printArray(const std::vector<int>& arr) {
    for (int num : arr) {
        std::cout << num << " ";
    }
    std::cout << std::endl;
}

// 主函数
int main() {
    std::vector<int> arr = {12, 11, 13, 5, 6, 7};
    std::cout << "Given array is \n";
    printArray(arr);

    mergeSort(arr, 0, arr.size() - 1);

    std::cout << "Sorted array is \n";
    printArray(arr);

    return 0;
}
  1. 二分查找:二分查找的时间复杂度为 O(log n),其中 n 是数组的大小。这种算法在处理大数据集时非常高效,尤其是在有序数据集上查找元素时。
#include <iostream>
#include <vector>

// 二分查找函数
int binarySearch(const std::vector<int>& arr, int left, int right, int target) {
    while (left <= right) {
        // 计算中间位置的索引
        int mid = left + (right - left) / 2;

        // 如果找到目标值,返回索引
        if (arr[mid] == target) {
            return mid;
        }
        // 如果目标值小于中间元素,更新右边界
        else if (target < arr[mid]) {
            right = mid - 1;
        }
        // 如果目标值大于中间元素,更新左边界
        else {
            left = mid + 1;
        }
    }
    // 如果没有找到目标值,返回-1
    return -1;
}

// 主函数
int main() {
    std::vector<int> arr = {2, 3, 4, 10, 40};
    int target = 10;

    // 调用二分查找函数
    int resultIndex = binarySearch(arr, 0, arr.size() - 1, target);

    // 输出结果
    if (resultIndex != -1) {
        std::cout << "Element found at index " << resultIndex << std::endl;
    } else {
        std::cout << "Element not found in the array." << std::endl;
    }

    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/497230.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

24/03/28总结

抽象类&#xff1a; 将共性的方法抽取到父类之后。由于每一个子类执行的内容是不一样&#xff0c;所以&#xff0c;在父类中不能确定具体的方法体。该方法就可以定义为抽象方法。 而为什么不直接在子类中定义方法&#xff1a;项目的完成不是一个人&#xff0c;如果有时忘记写方…

【教学类-40-09】A4骰子纸模制作9.0(3.47CM嵌套骰子 一条8格便于对折,表格相连 一页3个 油墨打印A4铅画纸)

作品展示 背景需求&#xff1a; 骰子调整到第8版&#xff0c;把骰子图案作成一长条&#xff0c;便于切割裁剪。 【教学类-40-08】A4骰子纸模制作8.0&#xff08;2.97CM嵌套骰子表格相连 一页7个 油墨打印A4铅画纸&#xff09;-CSDN博客文章浏览阅读929次&#xff0c;点赞20次…

幻兽帕鲁服务器价格表_阿里云/腾讯云/京东云/华为云报价大全

2024年全网最全的幻兽帕鲁服务器租用价格表&#xff0c;阿里云幻兽帕鲁游戏服务器26元1个月、腾讯云32元一个月、京东云26元一个月、华为云24元1个月&#xff0c;阿腾云atengyun.com整理最新幻兽帕鲁专用4核16G、8核16G、8核32G游戏服务器租用价格表大全&#xff1a; 阿里云幻…

土壤有机质空间分布数据

土壤有机质&#xff08;soil organic matter&#xff09;是土壤中含碳有机化合物的总称&#xff0c;包括土壤固有的和外部加入的所有动植物残体及其分解产物和合成产物。主要来源于动植物及微生物残体&#xff0c;可分为腐殖质和非腐殖物质。一般占土壤固相总重的10%以下&#…

推荐:便宜幻兽帕鲁Palworld联机游戏服务器优惠价格表

2024年全网最全的幻兽帕鲁服务器租用价格表&#xff0c;阿里云幻兽帕鲁游戏服务器26元1个月、腾讯云32元一个月、京东云26元一个月、华为云24元1个月&#xff0c;阿腾云atengyun.com整理最新幻兽帕鲁专用4核16G、8核16G、8核32G游戏服务器租用价格表大全&#xff1a; 阿里云幻…

思维升级之路:观察思维深层,解锁认知新境界

目录 一、观察我们的思维习惯 二、人类有哪些思维方法 三、为什么要多使用归纳推理而不是演绎推理 四、转变思维的关键 - 觉察 怎么提升自身的认知水平&#xff1f;这篇文章里&#xff0c;作者尝试对思维模式这件事做出探讨&#xff0c;一起来看看&#xff0c;或许可以帮你…

2023年蓝桥杯省赛——矩形面积总和

目录 题目链接&#xff1a; 1.矩形总面积 - 蓝桥云课 (lanqiao.cn) 思路 暴力 数学杯思路 数学逻辑 难点之重合区域 代码实现 总结 题目链接&#xff1a; 1.矩形总面积 - 蓝桥云课 (lanqiao.cn) 思路 暴力 开幕雷击&#xff0c;我直接开始暴力&#xff0c;但是我暴力…

Java学习之方法

目录 方法 方法声明格式&#xff1a; 调用方式&#xff1a; 详细说明 示例 --方法的声明及调用 语句块 练习 方法的重载(overload) 构成条件 示例 --方法重载 递归结构 缺陷 方法 方法(method)&#xff1a;一段用于完成特定功能的代码片段&#xff0c;类似于其他语…

Redis命令-List命令

4.6 Redis命令-List命令 Redis中的List类型与Java中的LinkedList类似&#xff0c;可以看做是一个双向链表结构。既可以支持正向检索和也可以支持反向检索。 特征也与LinkedList类似&#xff1a; 有序元素可以重复插入和删除快查询速度一般 常用来存储一个有序数据&#xff…

算法系列--动态规划--背包问题(2)--01背包拓展题目

&#x1f495;"2024.3.28小米汽车发布"&#x1f495; 作者&#xff1a;Lvzi 文章主要内容&#xff1a;算法系列–动态规划–背包问题(2)–01背包拓展题目 大家好,今天为大家带来的是算法系列--动态规划--背包问题(2)--01背包拓展题目 1.分割等和⼦集 链接: https:/…

能够解析任何编程语言的开源语法解析树 | 开源日报 No.171

tree-sitter/tree-sitter Stars: 14.6k License: MIT tree-sitter 是一个用于编程工具的增量解析系统。 该项目的主要功能、关键特性、核心优势包括&#xff1a; 通用性&#xff0c;能够解析任何编程语言高效性&#xff0c;能够在文本编辑器中每次按键都进行解析健壮性&…

Mysql的日志管理,备份与回复

目录 一、Mysql日志管理 1、日志的默认位置及配置文件 2、日志分类 2.1错误日志 2.2通用查询日志 2.3二进制日志 2.4慢查询日志 2.5中继日志 3、日志配置 4、日志查询 4.1查询通用日志是否开启 4.2查询二进制日志是否开启 4.3查看慢查询日志是否开启 4.4查询慢查…

web——rce,代码执行,命令执行

eval就会将里面的内容当成php来执行 ctf 第一 第二 过滤system\ 也可也怎么做 然后访问2.txt 第三&#xff08;参数逃逸&#xff09; 可以用这个&#xff0c;url中的eval是让get函数的使用&#xff0c;网页中的eval是为了让system中的函数起效 第四 过滤分号&#xff0c;因为上…

解决:PytorchStreamWriter failed writing file data

文章目录 问题内容问题分析解决思路 问题内容 今天在炼丹的时候&#xff0c;我发现模型跑到140步的时候保存权重突然报了个问题&#xff0c;详细内容如下&#xff1a; Traceback (most recent call last):File "/public/home/dyedd/.conda/envs/diffusers/lib/python3.8…

【Java核心能力】一篇文章了解 ZooKeeper 底层运行原理

欢迎关注公众号&#xff08;通过文章导读关注&#xff1a;【11来了】&#xff09;&#xff0c;及时收到 AI 前沿项目工具及新技术的推送&#xff01; 在我后台回复 「资料」 可领取编程高频电子书&#xff01; 在我后台回复「面试」可领取硬核面试笔记&#xff01; 文章导读地址…

Mysql数据库——主从复制与读写分离

目录 前言 一、主从复制 1.主从复制的定义 2.Mysql主从复制支持的类型 3.主从复制的过程 4. 主从复制出现的问题 5.解决方法 二、读写分离 1.读写分离的定义 2.读写分离的作用 3.读写分离作用场景 3.1基于程序代码内部实现 3.2基于中间代理层实现 4.主从复制与读…

Redis命令介绍

一、redis启动&#xff1a; 本地启动&#xff1a;redis-cli 远程启动&#xff1a;redis-cli -h host -p port -a password Redis 连接命令 1 AUTH password 验证密码是否正确 2 ECHO message 打印字符串 3 PING 查看服务是否运行 4 QUIT 关闭当前连接 5 SELECT index 切换…

UI设计案例,B端后台界面设计教程

B端产品是为“组织”提供服务&#xff0c;以业务为中心&#xff0c;追求时效性&#xff0c;在视觉上&#xff0c;内容为王&#xff0c;视觉为功能让步&#xff0c;追求简洁、清晰、克制、理性的视觉风格。B 端产品业务比较复杂&#xff0c;页面内容也会较多&#xff0c;B端界面…

Python与人工智能:气象领域的数据处理与模型优化

Python是功能强大、免费、开源&#xff0c;实现面向对象的编程语言&#xff0c;在数据处理、科学计算、数学建模、数据挖掘和数据可视化方面具备优异的性能&#xff0c;这些优势使得Python在气象、海洋、地理、气候、水文和生态等地学领域的科研和工程项目中得到广泛应用。可以…

LLM资料:中文embedding库

Highlight&#xff08;重点提示&#xff09; 理解LLM&#xff0c;就要理解Transformer&#xff0c;但其实最基础的还是要从词的embedding讲起。 毕竟计算机能处理的只有数字&#xff0c;所以万事开头的第一步就是将要处理的任务转换为数字。 面向中文的开源embedding库在自然…