算法——分治

学习目标:

  • 掌握算法入门知识

学习内容:

  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;
}
相关推荐
我是why的狗几秒前
赵义弘-----补题报告
算法·排序算法
roman_日积跬步-终至千里1 小时前
【机器学习】两大线性分类算法:逻辑回归与线性判别分析:找到分界线的艺术
算法·机器学习·分类
YuTaoShao1 小时前
【LeetCode 热题 100】739. 每日温度——(解法一)单调栈+从右到左
java·算法·leetcode·职场和发展
Spider_Man1 小时前
栈中藏玄机:从温度到雨水,单调栈的逆袭之路
javascript·算法·leetcode
爱吃KFC的大肥羊2 小时前
C/C++常用字符串函数
c语言·数据结构·c++·算法
im_AMBER2 小时前
Leetcode 11 java
java·算法·leetcode
丶小鱼丶2 小时前
二叉树算法之【Z字型层序遍历】
java·算法
武文斌773 小时前
嵌入式——数据结构:基础知识和链表①
数据结构
岁忧4 小时前
(nice!!!)(LeetCode 每日一题) 2561. 重排水果 (哈希表 + 贪心)
java·c++·算法·leetcode·go·散列表
先做个垃圾出来………4 小时前
1. 两数之和
算法·leetcode·职场和发展