news 2026/5/1 17:01:14

暴力枚举法解决环形石子合并问题

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
暴力枚举法解决环形石子合并问题

暴力枚举法解决环形石子合并问题:原理、代码与分析
一、问题引入
石子合并问题是区间动态规划的经典案例,而 “环形石子合并” 是其进阶形式:
在圆形操场的四周有 n 堆石子,每次只能合并相邻的两堆,合并得分是新堆的石子数。求将所有石子合并成一堆的最小得分和最大得分。
二、暴力枚举法的核心思路
暴力枚举法的本质是穷举所有可能的合并顺序,计算每种顺序的得分,最终取极值。
由于石子是环形排列,我们需要先将其转化为线性数组(拼接原数组,覆盖所有环形起点),再对每个线性子数组枚举所有合并顺序:
环形转线性:将原数组 stones 拼接一份(如 [4,1,2,3] → [4,1,2,3,4,1,2,3]),枚举所有长度为 n 的子数组,等价于枚举环形的所有起点。
枚举合并顺序:对每个线性子数组,递归枚举所有 “合并相邻堆” 的可能顺序,累计得分,最终记录全局最小 / 最大值。
三、C++ 代码实现(详细注释)
cpp
运行
#include <iostream>
#include <vector>
#include <climits>
#include <algorithm>
using namespace std;

// 全局变量:记录全局最小/最大得分
int global_min = INT_MAX;
int global_max = INT_MIN;

/**
* 递归暴力枚举所有合并顺序
* @param stones 当前剩余的石子堆数组
* @param current_score 当前累计的合并得分
*/
void bruteForceMerge(vector<int>& stones, int current_score) {
// 递归终止条件:只剩1堆石子,更新全局得分
if (stones.size() == 1) {
global_min = min(global_min, current_score);
global_max = max(global_max, current_score);
return;
}

// 枚举所有相邻的两堆,尝试合并
for (int i = 0; i < stones.size() - 1; ++i) {
// 1. 记录原始值(用于回溯)
int a = stones[i], b = stones[i + 1];
int merge_score = a + b; // 本次合并的得分

// 2. 原地合并:修改i位置,删除i+1位置
stones[i] = merge_score;
stones.erase(stones.begin() + i + 1);

// 3. 递归处理合并后的新数组
bruteForceMerge(stones, current_score + merge_score);

// 4. 回溯:恢复数组状态(保证其他分支的独立性)
stones.insert(stones.begin() + i + 1, b);
stones[i] = a;
}
}

int main() {
int n;
cout << "请输入石子堆数 n:";
cin >> n;

vector<int> stones(n);
cout << "请输入 " << n << " 堆石子的数量:";
for (int i = 0; i < n; ++i) {
cin >> stones[i];
}

// 环形转线性:枚举所有起点(覆盖环形的所有可能起始位置)
for (int start = 0; start < n; ++start) {
vector<int> linear_stones;
for (int i = 0; i < n; ++i) {
// 取模实现环形遍历,start为起点,i为偏移量
linear_stones.push_back(stones[(start + i) % n]);
}
// 对当前线性子数组枚举所有合并顺序
bruteForceMerge(linear_stones, 0);
}

// 输出结果
cout << "暴力枚举法最小得分:" << global_min << endl;
cout << "暴力枚举法最大得分:" << global_max << endl;

return 0;
}
四、代码运行示例
输入:
plaintext
请输入石子堆数 n:4
请输入 4 堆石子的数量:4 1 2 3
输出:
plaintext
暴力枚举法最小得分:19
暴力枚举法最大得分:26
五、暴力枚举法的优缺点分析
优点
逻辑直观:完全贴合 “合并相邻堆” 的规则,结果绝对正确,适合理解问题本质;
实现简单:无需复杂的动态规划或分治思想,仅需递归 + 枚举即可完成。
缺点
时间复杂度极高:
O(n×n!)
(n 为石子堆数),n=6 时需枚举约 720 种合并顺序,n=7 时需 5040 种,n>8 时几乎无法运行;
实用性差:仅能处理极小规模用例,无法应用于实际场景。
六、暴力法的优化方向
暴力法的核心问题是 “重复计算” 和 “阶乘级复杂度”,实际应用中需放弃暴力枚举,改用更高效的算法:
记忆化搜索:缓存 “合并区间 [i,j] 的得分”,时间复杂度降至 O(n 3);
区间动态规划:自底向上计算区间最优解,是环形石子合并的标准最优解法,时间复杂度O(n 3 ),可处理 n≤200 的大规模用例。
七、总结
暴力枚举法是理解 “石子合并问题” 的入门方式,但其阶乘级的时间复杂度决定了它仅适用于学习阶段。实际开发中,我们通常会使用区间动态规划来解决环形石子合并问题,既保证效率,又能得到全局最优解。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/29 0:50:53

3、Sentinel

随着微服务的流行&#xff0c;服务和服务之间的稳定性变得越来越重要。Sentinel 以流量为切入点&#xff0c;从{yellow}流量控制、熔断降级、系统负载保护等多个维度保护服务的稳定性。定义{green}资源&#xff1a; 主流框架{orange}自动适配{orange}编程式&#xff1a;SphU AP…

作者头像 李华
网站建设 2026/5/1 10:46:19

还在裸奔运行低代码PHP?立即实施这4步权限闭环策略

第一章&#xff1a;还在裸奔运行低代码PHP&#xff1f;权限闭环势在必行在现代Web开发中&#xff0c;低代码PHP脚本因其快速部署和简易维护被广泛使用&#xff0c;但若缺乏完善的权限控制机制&#xff0c;极易成为系统安全的突破口。许多开发者习惯于直接暴露功能性PHP文件&…

作者头像 李华
网站建设 2026/4/25 13:45:24

掌握这4种初始化模式,轻松玩转R量子计算模拟包

第一章&#xff1a;掌握R量子计算模拟包的qubit初始化核心概念在R语言的量子计算模拟环境中&#xff0c;正确理解与实现量子比特&#xff08;qubit&#xff09;的初始化是构建任何量子算法的基础。qubit作为量子信息的基本单元&#xff0c;其状态可表示为|0⟩和|1⟩的线性叠加。…

作者头像 李华
网站建设 2026/5/1 6:27:00

【资深架构师亲授】:大型项目中行为树的最佳实践与避坑策略

第一章&#xff1a;行为树的设计行为树&#xff08;Behavior Tree&#xff09;是一种用于建模智能体决策逻辑的树状结构&#xff0c;广泛应用于游戏AI、机器人控制和自动化系统中。其核心思想是将复杂的行为分解为可复用、可组合的节点&#xff0c;通过定义节点间的执行顺序和反…

作者头像 李华
网站建设 2026/5/1 4:56:44

多模态验证性能优化秘籍,Laravel 13高并发场景下的数据守护神

第一章&#xff1a;Laravel 13 多模态验证概述随着现代Web应用对数据输入来源的多样化&#xff0c;传统的单一表单验证机制已难以满足复杂场景的需求。Laravel 13 引入了多模态验证&#xff08;Multi-modal Validation&#xff09;机制&#xff0c;支持从API请求、表单提交、命…

作者头像 李华