Sticks<DFS>

题意:

给n个木棍,这些木棍是由m个长度均为L的木棍切割而来,求L的最小值。

思路:

DFS+剪枝。

剪枝:

1:L的取值范围在n(max)和n(sum)之间,逐个枚举。sum%L!=0则L不能用。
2:sort对n个木棍长度进行由小到大排序,有以下好处:
    a:从大的开始搜索。eg:如果L8;5+3>(优势)5+2+1;把更灵活地木棍留下来。
    b:查找更有序。
3:如果有一个木棍无法和其他木棍组合成L,则此种方案不行。<有两种情况>。
4:如果len[i]无法和当前的木棍组成,那么如果len[i+1]=len[i],len[i+1]也不行。
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
int len[5005],vis[5005];
int n,sum,num,L;
int cmp(int a,int b)
{
    return a>b;
}
int DFS(int m,int S,int N)//m代表当前已经组成的木棍数,S代表当前木棍的长度,N等会
{
    if(m==num-1)//b:已经有num-1根木棍组合好了,那么剩下的木棍一定也可以组成一根;
        return 1;
    for(int i=N;i<n;i++){
        if(!vis[i]&&(S+len[i])==L)
        {
            vis[i]=1;
            if(DFS(m+1,0,0))
                return 1;
            vis[i]=0;//虽然上步已经说明这条路行不通,还是要回溯回来原因是下一个else
            return 0;//C;
        }
        if(!vis[i]&&(S+len[i])<L)
        {
            vis[i]=1;
            if(DFS(m,S+len[i],i+1))
                return 1;
            vis[i]=0;
            if(S==0)//说明剩下的木棍无法组成;
                return 0;//D;
            while(len[i]==len[i+1])//E:此路不通,len[i]不行,如果len[i+1]=len[i],len[i+1]也不行。
                ++i;
        }
    }
    return 0;
}
int main()
{
    while(scanf("%d",&n)!=EOF)
    {
        if(n==0)
            break;
        sum=0;
        for(int i=0;i<n;i++){
            scanf("%d",&len[i]);
            sum+=len[i];
        }
        sort(len,len+n,cmp);
        for(L=len[0];L<=sum/2;L++)//a:减少L的枚举数量,如果L>sum/2,那么只能只有一根木棍
        {
            if(sum%L==0)//满足L的条件再递归
            {
                memset(vis,0,sizeof(vis));
                num=sum/L;
                if(DFS(0,0,0)){
                    printf("%d\n",L);
                    break;
                }
            }
        }
        if(L>sum/2)
            printf("%d\n",sum);
    }
    return 0;
}
### 关于第十五届蓝桥杯C++ C组拼正方形问题的代码实现 以下是针对第十五届蓝桥杯C++ C组拼正方形问题的一个可能的解决方案。此代码基于贪心算法的思想来分配木棍长度以形成尽可能多的正方形。 #### 代码实现 ```cpp #include <iostream> #include <vector> #include <algorithm> using namespace std; bool dfs(vector<int>& sticks, vector<int>& edges, int index, int edgeLength) { if (index == sticks.size()) { for (int i = 0; i < 4; ++i) { if (edges[i] != edgeLength) return false; } return true; } for (int i = 0; i < 4; ++i) { if (edges[i] + sticks[index] > edgeLength) continue; if (i > 0 && edges[i] == edges[i - 1]) continue; edges[i] += sticks[index]; if (dfs(sticks, edges, index + 1, edgeLength)) return true; edges[i] -= sticks[index]; if (edges[i] == 0) return false; } return false; } bool canFormSquare(vector<int> sticks) { sort(sticks.begin(), sticks.end(), greater<int>()); int totalLength = accumulate(sticks.begin(), sticks.end(), 0); if (totalLength % 4 != 0 || sticks.empty() || sticks.back() > totalLength / 4) return false; vector<int> edges(4, 0); return dfs(sticks, edges, 0, totalLength / 4); } int main() { vector<int> sticks = {1, 3, 4, 5, 7}; // 输入样例,可以根据题目调整输入数据 if (canFormSquare(sticks)) { cout << "可以拼成正方形" << endl; } else { cout << "无法拼成正方形" << endl; } return 0; } ``` 上述代码实现了通过DFS(深度优先搜索)方法尝试将给定的一系列木棒组合成四个边长相等的部分,从而构成一个正方形[^3]。程序首先对所有木棒按降序排列以便更快收敛,并利用剪枝技术减少不必要的计算路径。 #### 解决方案说明 该问题的核心在于如何合理地分割一组整数使之能够恰好组成四条相等长度的线段。这里采用了回溯法配合优化策略如排序和跳过重复状态等方式提高效率。具体而言: - **预处理阶段**: 计算总长度并验证其是否能被均匀分成四份;同时剔除任何单根超过目标边长的情况。 - **核心逻辑**: 使用递归函数`dfs()`探索每种可能性直到找到满足条件的结果或者穷尽所有选项。 这种方法虽然简单直观却也足够高效应对竞赛环境下的时间限制需求。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值