洛谷P4907【CYH-01】小奔的国庆练习赛 :换 (DFS,剪枝)
顺便提一下题意有一个地方不太清楚,就是如果输出No
还要输出最少需要添加多少张牌才能满足要求。蒟蒻考完以后发现四个点Too short on line 2
。。。
比较需要技巧的搜索
既然是同一个花色要连续,那就枚举每一个花色在哪一段区间连续并选中四个区间,累计每个点数的选中次数。
最后来一个的,首先每个点数选中次数要不少于已有的个数。接着,只有所有点数的选中次数和已有点数相等时,才能判为'Yes',然后统计某张牌的花色的区间未包含这张牌的总数更新答案。否则判为'No',然后统计每个点数选多了的总数更新答案。
总复杂度,超过了,肯定需要剪枝。
可行性剪枝:每个点数选中次数之和不少于原有牌的总数。
最优性剪枝:实时统计选多了的总数,在No
的状态下,如果超过答案则剪掉;在Yes
的状态下,只要不为就剪掉。
700+ms比标程快多了。可能标程比较良心没加什么剪枝。
#include<bits/stdc++.h>
#define R register int
using namespace std;
const int N=99;
int a[N],b[N],l[N],r[N],cnt[N],n,now1,ans=N,ans1=N;
char s[N];
void dfs(R h,R lef){//lef为n-当前已选中总次数
if(h==5){
R now=0;
for(R i=1;i<=13;++i){
if(cnt[i]>0)return;//不合法
now|=cnt[i]<0;
}
if(now){ans1=now1;return;}//No状态
for(R i=1;i<=n;++i)//Yes状态,统计答案
if((l[a[i]]>b[i]||r[a[i]]<b[i])&&++now==ans)return;
ans=now;ans1=1;return;//注意ans1=1的剪枝作用
}
for(R i=max(lef-(4-h)*13,0),j,rr;i<=13;++i){//枚举区间长度,可行性剪枝
if(i==0){l[h]=r[h]=0;dfs(h+1,lef);continue;}
for(rr=i;rr<=13;++rr){//枚举右端点
for(j=rr-i+1;j<=rr;++j)now1+=--cnt[j]<0;//动态维护当前选多了的总数
if(now1<ans1)l[h]=(r[h]=rr)-i+1,dfs(h+1,lef-i);//最优性剪枝
for(j=rr-i+1;j<=rr;++j)now1-=++cnt[j]<=0;
}
}
}
int main(){
scanf("%d",&n);
for(R i=1;i<=n;++i){
scanf("%d%s",&a[i],s);
if(s[0]=='A')b[i]=1;//把点数处理一下
else if(s[0]=='1')b[i]=10;
else if(s[0]=='J')b[i]=11;
else if(s[0]=='Q')b[i]=12;
else if(s[0]=='K')b[i]=13;
else b[i]=s[0]-'0';
++cnt[b[i]];
}
dfs(1,n);
if(ans!=N)printf("Yes\n%d\n",ans);
else printf("No\n%d\n",ans1);
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 开发者必知的日志记录最佳实践
· SQL Server 2025 AI相关能力初探
· Linux系列:如何用 C#调用 C方法造成内存泄露
· AI与.NET技术实操系列(二):开始使用ML.NET
· 记一次.NET内存居高不下排查解决与启示
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· AI技术革命,工作效率10个最佳AI工具