算法——分治

学习目标:

  • 掌握算法入门知识

学习内容:

  1. 分治的定义
  2. 例题详细步骤讲解(查找最大和次大元素)

1. 分治的定义

对于一个大规模的问题,将其分解为k个规模较小的子问题,这些子问题互相独立且与原问题形式相同,递归 地解这些子问题,然后将各子问题的解合并得到原问题的解。

① 分解:将原问题分解为若干个规模较小且与原问题形式相同的子问题。

② 求解子问题:若子问题规模较小容易被解决则直接求解,否则递归地求解各个子问题。

③ 合并:将各个子问题的解合并为原问题的解。

2. 例题详细步骤讲解

查找最大和次大元素:对于给定的含有n元素的无序序列,求这个序列中最大和次大的两个不同的元素。例如:(2, 5, 1, 4, 6, 3),最大元素为6,次大元素为5。
  • 具体分析:

    对于一个无序序列a,假设最初 a[0]为a[low],a[n-1]为a[high],最大元素max1为0,次大元素max2为0,MAX(x,y)用来求两个元素中的最大值,MIN(x,y)用来求两个元素中的最小值:

    (1)当数组a中只有一个 元素:max1=a[low],max2=-INF。

    (2)当数组a中只有两个 元素:max1=MAX(a[low],a[high]),max2=MIN(a[low],a[high])。

    (3)当数组a中有两个以上元素:

    • 划分:按中间位置 mid=(low+high)/2 分为a[low...mid]和a[mid+1...high]左右两个区间。
    • 求出左区间最大元素lmax1和次大元素lmax2,求出右区间最大元素rmax1和次大元素rmax2。
    • 合并:若lmax1>rmax1,则max1=lmax1,max2=MAX(lmax2,rmax1);否则max1=rmax1,max2=MAX(lmax1,rmax2)。
  • 具体过程:

    (2,5,1,4,6,3)

low=0,high=5,high-low>1数组中有两个以上元素,进行划分。mid=(low+high)/2=2,递归求解a[0...2],a[3...5]}
(划分左区间a[0...2])low=0,high=2,high-low>1数组中有两个以上元素,进行划分。mid=(low+high)/2=1,递归求解a[0...1],a[2...2]

  • (划分左区间a[0...1])low=0,high=1,high-low=1数组中有两个元素。max1=5,max2=2
  • (划分右区间a[2...2])low=2,high=2,high-low=0数组中有1个元素。max1=a[low]=1,max2=-INF
  • 进行合并,在(左区间)中得到的分别为lmax1和lmax2,在(右区间)中得到的分别为rmax1和rmax2。得到该部分max1=5,max2=2

(划分右区间a[3...5])low=3,high=5,high-low>1数组中有两个以上元素,进行划分。mid=(low+high)/2=4,递归求解a[3...4],a[5...5]

  • (划分左区间a[3...4])low=3,high=4,high-low=1数组中有两个元素。max1=6,max2=4
  • (划分右区间a[5...5])low=5,high=5,high-low=0数组中有1个元素。max1=a[low]=3,max2=-INF
  • 进行合并,在(左区间)中得到的分别为lmax1和lmax2,在(右区间)中得到的分别为rmax1和rmax2。得到该部分max1=6,max2=4

进行合并,lmax1=5,lmax2=2,rmax1=6,rmax2=4。得出max1=6,max2=5

  • 具体代码:
c 复制代码
#include <stdio.h>
#include <limits.h> // 用于定义INT_MIN

#define INF INT_MIN

void solve(int a[], int low, int high, int *max1, int *max2) {
    if (low == high) {       // 区间只有一个元素
        *max1 = a[low];
        *max2 = -INF;
    } else if (low == high - 1) { // 区间只有两个元素
        *max1 = (a[low] > a[high]) ? a[low] : a[high];
        *max2 = (a[low] < a[high]) ? a[low] : a[high];
    } else {                // 区间有两个以上元素
        int mid = (low + high) / 2;
        int lmax1, lmax2;
        solve(a, low, mid, &lmax1, &lmax2);      // 左区间求lmax1和lmax2
        int rmax1, rmax2;
        solve(a, mid + 1, high, &rmax1, &rmax2); // 右区间求rmax1和rmax2
        
        if (lmax1 > rmax1) {
            *max1 = lmax1;
            *max2 = (lmax2 > rmax1) ? lmax2 : rmax1; // lmax2, rmax1中求次大元素
        } else {
            *max1 = rmax1;
            *max2 = (lmax1 > rmax2) ? lmax1 : rmax2; // lmax1, rmax2中求次大元素
        }
    }
}

int main() {
    int a[] = {2, 5, 1, 4, 6, 3};
    int n = sizeof(a) / sizeof(a[0]);
    int max1, max2;
    
    solve(a, 0, n - 1, &max1, &max2);
    
    printf("最大元素: %d\n", max1);
    printf("次大元素: %d\n", max2);
    
    return 0;
}
相关推荐
你撅嘴真丑3 小时前
第九章-数字三角形
算法
uesowys4 小时前
Apache Spark算法开发指导-One-vs-Rest classifier
人工智能·算法·spark
ValhallaCoder4 小时前
hot100-二叉树I
数据结构·python·算法·二叉树
董董灿是个攻城狮4 小时前
AI 视觉连载1:像素
算法
智驱力人工智能4 小时前
小区高空抛物AI实时预警方案 筑牢社区头顶安全的实践 高空抛物检测 高空抛物监控安装教程 高空抛物误报率优化方案 高空抛物监控案例分享
人工智能·深度学习·opencv·算法·安全·yolo·边缘计算
孞㐑¥5 小时前
算法——BFS
开发语言·c++·经验分享·笔记·算法
月挽清风5 小时前
代码随想录第十五天
数据结构·算法·leetcode
XX風5 小时前
8.1 PFH&&FPFH
图像处理·算法
NEXT066 小时前
前端算法:从 O(n²) 到 O(n),列表转树的极致优化
前端·数据结构·算法
代码游侠6 小时前
学习笔记——设备树基础
linux·运维·开发语言·单片机·算法