题目描述

牛牛最近迷上了一种叫斗地主的扑克游戏。斗地主是一种使用黑桃、红心、梅花、方片的A到K加上大小王的共54张牌来进行的扑克牌游戏。在斗地主中,牌的大小关系根据牌的数码表示如下:3<4<5<6<7<8<9<10<J<Q<K<A<2<小王<大王,而花色并不对牌的大小产生影响。每一局游戏中,一副手牌由n张牌组成。游戏者每次可以根据规定的牌型进行出牌,首先打光自己的手牌一方取得游戏的胜利。

现在,牛牛只想知道,对于自己的若干组手牌,分别最少需要多少次出牌可以将它们打光。请你帮他解决这个问题。

需要注意的是,本题中游戏者每次可以出手的牌型与一般的斗地主相似而略有不同。

具体规则如下:

输入输出格式

输入格式:

第一行包含用空格隔开的2个正整数T和n,表示手牌的组数以及每组手牌的张数。

接下来T组数据,每组数据n行,每行一个非负整数对aibi表示一张牌,其中ai示牌的数码,bi表示牌的花色,中间用空格隔开。特别的,我们用1来表示数码A,11表示数码J,12表示数码Q,13表示数码K;黑桃、红心、梅花、方片分别用1-4来表示;小王的表示方法为01,大王的表示方法为02。

输出格式:

共T行,每行一个整数,表示打光第i手牌的最少次数。

输入输出样例

输入样例#1:

1 8
7 4
8 4
9 1
10 4
11 1
5 1
1 4
1 1
输出样例#1:

3
输入样例#2:

1 17
12 3
4 3
2 3
5 4
10 2
3 3
12 2
0 1
1 3
10 1
6 2
12 1
11 3
5 2
12 4
2 2
7 2
输出样例#2:

6

说明

样例1说明

共有1组手牌,包含8张牌:方片7,方片8,黑桃9,方片10,黑桃J,黑桃5,方片A以及黑桃A。可以通过打单顺子(方片7,方片8,黑桃9,方片10,黑桃J),单张牌(黑桃5)以及对子牌(黑桃A以及方片A)在3次内打光。

对于不同的测试点, 我们约定手牌组数T与张数n的规模如下:

数据保证:所有的手牌都是随机生成的。

尼玛广搜323行85分,深搜压行之后57行就A了,,,‘

 #include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
using namespace std;
const int MAXN=;
int read(int & n)
{
char c='-';int x=;
while(c<''||c>'')c=getchar();
while(c>=''&&c<='')
{
x=x*+(c-);
c=getchar();
}
n=x;
}
int T,n,p,hs;
int ans;
int card_num[MAXN];// 记录每一种数码的出现次数
int happen[MAXN];// 记录数量的出现次数
/*比如说3出现了两次,A出现了两次,那么happen[2]==2*/
int take_num[]={,,,};// 斗地主的规则,分别对应单顺双顺三顺
void dfs(int now)// now是指已经操作的次数
{
if(now>ans)
return ;// 剪枝
memset(happen,,sizeof(happen));
for(int i=;i<=;i++)
happen[card_num[i]]++;
int cs=;// 本轮的操作次数
while(happen[])// 四带
{
cs++;
happen[]--;
if(happen[]>=)//根据贪心的原理,能出两张则不出一张
happen[]-=;// 能带两套对牌不带一套对牌
else if(happen[]>=)
happen[]-=;//四张牌每次可以带两张单牌
}
while(happen[])
{
cs++;
happen[]--;
if (happen[])
happen[]--;
else if(happen[])
happen[]--;//思路同上,三张牌进行带牌的时候只能带一张
}
if(card_num[]&&card_num[]&&happen[]>=)
cs--;//当大王和小王可以同时出的时候就当做对牌一起出
// 因为在后面一条语句中需要+happen[1],所以默认是大王小王当单牌出
// 那么同时有大王小王就需要两次操作,实际上一次操作就可以完成,相当于2-1=1
cs=cs+happen[]+happen[];
// 剩下的对牌和单牌需要每组一次操作
ans=min(ans,cs+now);// 更新答案
for(int k=;k<=;k++)// k代表顺子的类型,1:单顺 2:双顺 3:三顺
{
for(int i=,j;i<=;++i)// 枚举每一张牌,因为2不能在顺子中出现,所以无视
{
for(j=i;card_num[j]>=k&&j<=;++j)
{//在可行的情况和区间内寻找顺子
card_num[j]-=k;// 先减去,后面会加回来
if(j-i+>=take_num[k])// 可以当顺子出
dfs(now+);// 就当顺子出
}
while(j>i)// 递归的回溯
card_num[--j]+=k;
}
} }
int main()
{
read(T);read(n);
while(T--)
{
memset(card_num,,sizeof(card_num));
ans=n;
for(int i=;i<=n;i++)
{
read(p);read(hs);
if(p==)card_num[hs-]++;
// 把小王看做0,大王看做1.保证card_num数组没有冲突
else if(p==)card_num[]++;// 把A看做14
else card_num[p]++;
} dfs();
printf("%d\n",ans);
}
return ;
}
 #include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;
const int MAXN=;
int read(int & n)
{char c='-';int x=;while(c<''||c>'')c=getchar();while(c>=''&&c<='') {x=x*+(c-),c=getchar();}n=x;}
int T,n,p,hs,ans;
int card_num[MAXN],happen[MAXN],take_num[]={,,,};
void dfs(int now)// now是指已经操作的次数
{
if(now>ans) return ;// 剪纸
memset(happen,,sizeof(happen));
for(int i=;i<=;i++) happen[card_num[i]]++;
int cs=;// 本轮的操作次数
for(int i=;i<=;i++)
while(happen[+i])
{
cs++,happen[+i]--;
if (happen[]>=+i) happen[]-=+i;
else if(happen[]>=+i) happen[]-=+i;
}
if(card_num[]&&card_num[]&&happen[]>=) cs--;
cs=cs+happen[]+happen[];
ans=min(ans,cs+now);
for(int k=;k<=;k++)
for(int i=,j;i<=;++i)
{
for(j=i;card_num[j]>=k&&j<=;++j)
{
card_num[j]-=k;
if(j-i+>=take_num[k])
dfs(now+);
}
while(j>i)
card_num[--j]+=k;
}
}
int main()
{
read(T);read(n);
while(T--)
{
memset(card_num,,sizeof(card_num));
ans=n;
for(int i=;i<=n;i++)
{
read(p);read(hs);
if(p==)card_num[hs-]++;
else if(p==)card_num[]++;
else card_num[p]++;
}
dfs();
printf("%d\n",ans);
}
return ;
}

57

P2668 斗地主 贪心+深搜的更多相关文章

  1. P2668 斗地主 dp+深搜版

    题目描述 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游戏.在斗地主中,牌的大小关系根据牌的数码表示如下:3<4< ...

  2. Luogu P2668 斗地主(NOIP2015)

    还记得那道我只用特判得了30分的"斗地主"吗? 我今天脑抽打算把它改A掉.为什么不用这大好时光去干些更有意义的事 于是我就挖了这个坑. 题解: 题目链接:P2668 斗地主 本题就 ...

  3. 2015暑假多校联合---Cake(深搜)

    题目链接:HDU 5355 http://acm.split.hdu.edu.cn/showproblem.php?pid=5355 Problem Description There are m s ...

  4. 【BZOJ】1016: [JSOI2008]最小生成树计数 深搜+并查集

    最小生成树计数 Description 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小 ...

  5. 模拟赛T5 : domino ——深搜+剪枝+位运算优化

    这道题涉及的知识点有点多... 所以还是比较有意思的. domino 描述 迈克生日那天收到一张 N*N 的表格(1 ≤ N ≤ 2000),每个格子里有一个非 负整数(整数范围 0~1000),迈克 ...

  6. [noip模拟]小猫爬山<迭代深搜>

    [题目描述]: Freda和rainbow饲养了N只小猫,这天,小猫们要去爬山.经历了千辛万苦,小猫们终于爬上了山顶,但是疲倦的它们再也不想徒步走下山了(呜咕>_<). Freda和rai ...

  7. HDU--杭电--1195--Open the Lock--深搜--都用双向广搜,弱爆了,看题了没?语文没过关吧?暴力深搜难道我会害羞?

    这个题我看了,都是推荐的神马双向广搜,难道这个深搜你们都木有发现?还是特意留个机会给我装逼? Open the Lock Time Limit: 2000/1000 MS (Java/Others)  ...

  8. 利用深搜和宽搜两种算法解决TreeView控件加载文件的问题。

    利用TreeView控件加载文件,必须遍历处所有的文件和文件夹. 深搜算法用到了递归. using System; using System.Collections.Generic; using Sy ...

  9. 2016弱校联盟十一专场10.3---Similarity of Subtrees(深搜+hash、映射)

    题目链接 https://acm.bnu.edu.cn/v3/problem_show.php?pid=52310 problem description Define the depth of a ...

随机推荐

  1. how to read openstack code : paste deploy

    本篇分为以下几个部分 paste 是什么 怎样使用paste paste of neutron paste 是什么 WSGI 是python 中application 和 web server互通的标 ...

  2. linux命令dmesg查看进程被杀死原因

    有一次一个python进程挂了,使用了下这个命令. 可以看到原因:虚拟机总共内存8082608KB,结果python项目就消耗掉了7341764KB,内存泄露,导致python进程被系统杀死 顺带介绍 ...

  3. Growth: 一个关于怎样成为优秀Web Developer 的 App

    想了想还是决定在今天公布一个预览版.这样才干持续改进.Growth是一个关于怎样成为优秀的Web Developer的APP--结合技能树.成长路线图.进阶书单.Web七日谈以及一些小測验. 它是我对 ...

  4. 水晶报表 Crystal Report 调用存储过程时出错 找不到表 ,解决方法。

    用 CrystalReportViewer1 控件在asp.net的网页上显示报表,假设做报表时调用数据表数据的方式调用是能够成功的.但报表是用存储过程获取数据方式会出现下面错误: 找不到表'RptO ...

  5. HDU4930-Fighting the Landlords

    题意:斗地主,就是要自己出牌.使得对手在这一轮无法出牌,或者有出牌的可能.可是你的牌已经走完了.假设符合这些条件的话,输出Yes.否则输出No. 思路:先预处理能直接把牌走完的情况,假设不行的话就直接 ...

  6. solr入门之solr的拼写检查功能的应用级别尝试

    今天主要是收集了些拼写检查方面的资料和 尝试使用一下拼写检查的功能--=遇到了不少问题 拼写检查的四种配置眼下我仅仅算是成功了半个吧 --------------------------------- ...

  7. Java的Graphics中drawImage与drawLine的坐标区别

    drawImage复制的区域是 dx1 <= x < dx2,dy1 <= y < dy2 drawLine绘制区域是 dx1 <= x <= dx2,dy1 &l ...

  8. kentico7中设置site的default page

    在SiteManager中,Settings中,选择好站点,然后Content,Default alias path

  9. [Codeforces 140C] New Year Snowmen

    [题目链接] https://codeforces.com/problemset/problem/140/C [算法] 显然 , 我们每次应优先考虑数量多的雪球 将雪球个数加入堆中 , 每次取出数量前 ...

  10. python 统计文件的个数

    import os path = r'F:\1back\picture' #获取当前路径 count = 0 for root,dirs,files in os.walk(path): #遍历统计 i ...