题目描述

牛牛最近迷上了一种叫斗地主的扑克游戏。斗地主是一种使用黑桃、红心、梅花、方片的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. 洛谷 P2237 [USACO14FEB]自动完成Auto-complete

    P2237 [USACO14FEB]自动完成Auto-complete 题目描述 Bessie the cow has a new cell phone and enjoys sending text ...

  2. HAProxy教程收集

    市面上HA的教程不是很多,基本都是基于LVS+HA实践的打包资料. 要最权威的文档应该去官方. 官方文档入口: http://www.haproxy.org/#docs 中文文档收集: http:// ...

  3. Servlet中操作数据库

    以下内容引用自http://wiki.jikexueyuan.com/project/servlet/database-access.html: 前提先新建数据库及插入模拟数据: create tab ...

  4. 几点平时不太注意的CSS知识

    1:文本显示的时候,我们发现左右参差不齐,text-align:justify  就能让文本左右都齐刷刷的啦: 2:input标签的内容,处于安全考虑,有时候我们并不希望别人黏贴复制,这时候这样干:& ...

  5. xming + putty remote GUI

    xming 和putty的配置网上有很多 但是在使用时发现有个问题, 记录一下. 在配置完成后,远程运行图形化程序经常会说can not open display等错误. 这有可能是因为xming的安 ...

  6. java文件工具类

    import java.io.BufferedReader; import java.io.BufferedWriter; import java.io.File; import java.io.Fi ...

  7. centos7容量扩充

    新买的2T 绿盘到货了~~好开心的说~但毕竟是第一次安装,事先还是在网上搜索了很多资料才敢动手,下面就开始啦~ 环境:Centos7.dell服务器.2T容量绿盘 1.硬盘连接好之后,开机先使用fdi ...

  8. C# 如何修改Form不能修改窗体大小

    把窗体的FormBorderSytle改一下就可以了,改成FixedSingle或者Fixed3D都可以        

  9. 86. LotusScript中的数组函数

    R6对LotusScript有一些改进和增强,自那之后.Notes对象的接口时有补充和更新,但语言本身没有变化.那些改进就包括添加诸如ArrayGetIndex.ArrayUnique的实用函数. 但 ...

  10. UVA1523-Helicopter(暴力+全排列)

    题目链接 题意:有八个乘客坐在直升机上,求重心M最小值. 思路:依据题目所给的公式,我们能够知道要使得M最小.也就是要使得Mv和Mh的和最小,我们能够使用全排列,分别将每一个值放在各个位子上,然后更新 ...