算法之分治算法
换句话说,将大问题分解为多个子问题、解决子问题、将子问题的解合并为原问题的解,这几步的效率为什么比直接解决原问题的效率更高?:快速排序是选取一个基准值,然后把数组分为两个子数组,一个子数组的元素比基准值小另一子数组的元素比基准值大,然后再对这两部分进行相同的划分操作,直至子数组只剩下一个元素。:桶排序的基本思想是将数据分散到多个桶,然后对每个桶内的元素进行排序,最后将各个桶的元素依次取出,从而得到
1.分治算法
1.1如何判断分治问题
1.2 通过分治提升效率
1. 操作数量优化以“冒泡排序”为例,其处理一个长度为 𝑛 的数组需要𝑂(𝑛的2次方 ) 时间。假设我们按照下图 所示的方式,将数组从中点分为两个子数组,则划分需要 𝑂(n) 时间,排序每个子数组需要𝑂((𝑛/2)2 ) 时间,合并两个子数组需要 𝑂(𝑛) 时间,总体时间复杂度为:![]()
再思考,如果我们多设置几个划分点,将原数组平均划分为k个子数组呢?这种情况与“桶排序”非常类似,它非常适合排序海量数据,理论上时间复杂度可以达到O(n + k)。
2. 并行计算优化我们知道,分治生成的子问题是相互独立的,因此通常可以并行解决。也就是说,分治不仅可以降低算法的时间复杂度,还有利于操作系统的并行优化。并行优化在多核或多处理器的环境中尤其有效,因为系统可以同时处理多个子问题,更加充分地利用计算资源,从而显著减少总体的运行时间。比如在下图所示的“桶排序”中,我们将海量的数据平均分配到各个桶中,则可以将所有桶的排序任务分散到各个计算单元,完成后再进行结果合并。![]()
1.3 分治常见应用
2 分治搜索策略
2.1. 基于分治实现二分
Q:给定一个长度为 𝑛 的有序数组 nums ,数组中所有元素都是唯一的,请查找元素 target
/* 二分查找:问题 f(i, j) */
int dfs(int[] nums, int target, int i, int j) {
// 若区间为空,代表无目标元素,则返回 -1
if (i > j) { //校验传入数据是否有误
return -1;
}
// 计算中点索引 m
int m = (i + j) / 2;
if (nums[m] < target) {
// 递归子问题 f(m+1, j)
return dfs(nums, target, m + 1, j);
} else if (nums[m] > target) {
// 递归子问题 f(i, m-1)
return dfs(nums, target, i, m - 1);
} else {
// 找到目标元素,返回其索引
return m;
}
}
/* 二分查找 */
int binarySearchRecursion(int[] nums, int target) {
int n = nums.length;
// 求解问题 f(0, n-1)
return dfs(nums, target, 0, n - 1);
}
3 构建二叉树问题
Q:
给定一个二叉树的前序遍历 preorder 和中序遍历 inorder ,请从中构建二叉树,返回二叉树的根节点。假设二叉树中没有值重复的节点。
前序遍历(Preorder Traversal):
- 定义:从树的根节点出发,在递归遍历左子树和右子树的过程中,先访问根节点。
- 遍历顺序:根节点 -> 左子树 -> 右子树。
中序遍历(Inorder Traversal):
- 定义:从树的根节点出发,在递归遍历左子树和右子树的过程中,中间访问根节点。
- 遍历顺序:左子树 -> 根节点 -> 右子树。
后序遍历(Postorder Traversal):
- 定义:从树的根节点出发,在递归遍历左子树和右子树的过程中,最后访问根节点。
- 遍历顺序:左子树 -> 右子树 -> 根节点。
这三种遍历方式是针对树这种递归结构的,它们都可以用递归或迭代的方式实现。在实际应用中,选择不同的遍历方式取决于问题的要求。例如,前序遍历常用于复制整棵树,中序遍历常用于获取有序序列,后序遍历常用于释放树的节点等。
举例说明,以下二叉树:
1
/ \
2 3
/ \
4 5
- 前序遍历:1 -> 2 -> 4 -> 5 -> 3
- 中序遍历:4 -> 2 -> 5 -> 1 -> 3
- 后序遍历:4 -> 5 -> 2 -> 3 -> 1
1.判断是否为分治问题
2. 如何划分子树
3. 基于变量描述子树区间
4. 代码实现
为了提升查询 𝑚 的效率,我们借助一个哈希表 hmap 来存储数组 inorder 中元素到索引的映射。
import java.util.HashMap;
import java.util.Map;
public class Test {
public static void main(String[] args) {
}
/* 构建二叉树:分治 */
TreeNode dfs(int[] preorder, Map<Integer, Integer> inorderMap, int i, int l, int r) {
// 子树区间为空时终止
if (r - l < 0)
return null;
// 初始化根节点
TreeNode root = new TreeNode(preorder[i]);
// 查询 m ,从而划分左右子树
int m = inorderMap.get(preorder[i]);
// 子问题:构建左子树
root.left = dfs(preorder, inorderMap, i + 1, l, m - 1);
// 子问题:构建右子树
root.right = dfs(preorder, inorderMap, i + 1 + m - l, m + 1, r);
// 返回根节点
return root;
}
/* 构建二叉树 */
TreeNode buildTree(int[] preorder, int[] inorder) {
// 初始化哈希表,存储 inorder 元素到索引的映射
Map<Integer, Integer> inorderMap = new HashMap<>();
for (int i = 0; i < inorder.length; i++) {
inorderMap.put(inorder[i], i);
}
TreeNode root = dfs(preorder, inorderMap, 0, 0, inorder.length - 1);
return root;
}
}

每个递归函数内的前序遍历preorder和中序遍历inorder 的划分结果如下图所示。
设树的节点数量为 𝑛 ,初始化每一个节点(执行一个递归函数 dfs() )使用 𝑂(1) 时间。因此总体时间复杂 度为 𝑂(𝑛) 。
4.汉诺塔问题
给定三根柱子,记为 A、B 和 C 。起始状态下,柱子 A 上套着 𝑛 个圆盘,它们从上到下按照从小到大的顺序排列。我们的任务是要把这 𝑛 个圆盘移到柱子 C 上,并保持它们的原有顺序不变。在移动圆盘的过程中,需要遵守以下规则。
1. 圆盘只能从一个柱子顶部拿出,从另一个柱子顶部放入。2. 每次只能移动一个圆盘。3. 小圆盘必须时刻位于大圆盘之上。
我们将规模为 𝑖 的汉诺塔问题记做 𝑓(𝑖) 。例如 𝑓(3) 代表将 3 个圆盘从 A 移动至 C 的汉诺塔问题。
1. 考虑基本情况


解决问题 𝑓(2) 的过程可总结为:将两个圆盘借助 B 从 A 移至 C 。其中,C 称为目标柱、B 称为缓冲柱。
2.子问题分解


3.代码
/* 移动一个圆盘 */
void move(List<Integer> src, List<Integer> tar) {
// 从 src 顶部拿出一个圆盘
Integer pan = src.remove(src.size() - 1);
// 将圆盘放入 tar 顶部
tar.add(pan);
}
/* 求解汉诺塔:问题 f(i) */
void dfs(int i, List<Integer> src, List<Integer> buf, List<Integer> tar) {
// 若 src 只剩下一个圆盘,则直接将其移到 tar
if (i == 1) {
move(src, tar);
return;
}
// 子问题 f(i-1) :将 src 顶部 i-1 个圆盘借助 tar 移到 buf
dfs(i - 1, src, tar, buf);
// 子问题 f(1) :将 src 剩余一个圆盘移到 tar
move(src, tar);
// 子问题 f(i-1) :将 buf 顶部 i-1 个圆盘借助 src 移到 tar
dfs(i - 1, buf, src, tar);
}
/* 求解汉诺塔 */
void solveHanota(List<Integer> A, List<Integer> B, List<Integer> C) {
int n = A.size();
// 将 A 顶部 n 个圆盘借助 B 移到 C
dfs(n, A, B, C);
}
5 小结
更多推荐


所有评论(0)