贪心 DFS
DFS枚举顺子情况,然后剩下的贪心出牌。
详情见注释
有兴趣的小伙伴可以A一A洛谷的数据加强版。(我这个过不了)
代码:
#include<cstdio>
#include<cstring>
#include<algorithm>
#define MAXN 14
using namespace std;
int t,n,ans;
int num[MAXN+5];
int sum[MAXN+5];
int to[MAXN+5]={0,13,1,2,3,4,5,6,7,8,9,10,11,12};
int calc(){
memset(sum,0,sizeof(sum));//sum存是单牌/对子/三张牌/炸弹的数量
for (int i=0;i<=13;i++)
sum[num[i]]++;
int ret=0;
while (sum[4]>0&&sum[2]>1){//炸弹带对子最先,因为出的牌最多,下同
sum[4]--; sum[2]-=2; ret++;
}
while (sum[4]>0&&sum[1]>1){
sum[4]--; sum[1]-=2; ret++;
}
while (sum[4]>0&&sum[2]>0){
sum[4]--; sum[2]--; ret++;
}
while (sum[4]>0&&sum[1]>0){
sum[4]--; sum[1]--; ret++;
}
while(sum[4]>1){//两个炸弹可以拆成一个炸弹和两个对子,就可以一次出了(不过貌似不卡这个)
sum[4]-=2; ret++;
}
while (sum[3]>0&&sum[2]>0){
sum[3]--; sum[2]--; ret++;
}
while (sum[3]>0&&sum[1]>0){
sum[3]--; sum[1]--; ret++;
}
return ret+sum[1]+sum[2]+sum[3]+sum[4];//总和
}
void dfs(int sum){//DFS搜顺子
if (sum>ans) return;//如果当前次数已经比答案大了就退出
ans=min(ans,sum+calc());//更新答案
for (int i=2;i<=13;i++)
for (int j=1;j<=3;j++)//枚举顺子的形式
if (num[i]>=j){
int k=i,p=0;
while (num[k]>=j){
p+=j; num[k++]-=j;
}
k--;
while (k>=i){
if (p>=5)
dfs(sum+1);
num[k--]+=j; p-=j;
}
}
}
int main(){
scanf("%d%d",&t,&n);
while (t--){
memset(num,0,sizeof(num));
int x; ans=0x7fffffff;
for (int i=1;i<=n;i++){
scanf("%d%*d",&x);//%*即跳过不读
num[to[x]]++;
}
dfs(0);
printf("%d\n",ans);
}
}