洛谷P2668 斗地主(NOIp2015)(BZOJ4325)

贪心 DFS

洛谷题目传送门
BZOJ题目传送门

DFS枚举顺子情况,然后剩下的贪心出牌。
详情见注释

有兴趣的小伙伴可以A一A洛谷的数据加强版。(我这个过不了)

代码:

#include
#include
#include
#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);
    }
}

你可能感兴趣的:(BZOJ,洛谷,搜索---DFS(序),其他---贪心,蒟蒻zxl的Blog专栏)