P2668 斗地主 贪心+深搜
题目描述
牛牛最近迷上了一种叫斗地主的扑克游戏。斗地主是一种使用黑桃、红心、梅花、方片的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 8
7 4
8 4
9 1
10 4
11 1
5 1
1 4
1 1
3
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
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 斗地主 贪心+深搜的更多相关文章
- P2668 斗地主 dp+深搜版
题目描述 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游戏.在斗地主中,牌的大小关系根据牌的数码表示如下:3<4< ...
- Luogu P2668 斗地主(NOIP2015)
还记得那道我只用特判得了30分的"斗地主"吗? 我今天脑抽打算把它改A掉.为什么不用这大好时光去干些更有意义的事 于是我就挖了这个坑. 题解: 题目链接:P2668 斗地主 本题就 ...
- 2015暑假多校联合---Cake(深搜)
题目链接:HDU 5355 http://acm.split.hdu.edu.cn/showproblem.php?pid=5355 Problem Description There are m s ...
- 【BZOJ】1016: [JSOI2008]最小生成树计数 深搜+并查集
最小生成树计数 Description 现在给出了一个简单无向加权图.你不满足于求出这个图的最小生成树,而希望知道这个图中有多少个不同的最小生成树.(如果两颗最小生成树中至少有一条边不同,则这两个最小 ...
- 模拟赛T5 : domino ——深搜+剪枝+位运算优化
这道题涉及的知识点有点多... 所以还是比较有意思的. domino 描述 迈克生日那天收到一张 N*N 的表格(1 ≤ N ≤ 2000),每个格子里有一个非 负整数(整数范围 0~1000),迈克 ...
- [noip模拟]小猫爬山<迭代深搜>
[题目描述]: Freda和rainbow饲养了N只小猫,这天,小猫们要去爬山.经历了千辛万苦,小猫们终于爬上了山顶,但是疲倦的它们再也不想徒步走下山了(呜咕>_<). Freda和rai ...
- HDU--杭电--1195--Open the Lock--深搜--都用双向广搜,弱爆了,看题了没?语文没过关吧?暴力深搜难道我会害羞?
这个题我看了,都是推荐的神马双向广搜,难道这个深搜你们都木有发现?还是特意留个机会给我装逼? Open the Lock Time Limit: 2000/1000 MS (Java/Others) ...
- 利用深搜和宽搜两种算法解决TreeView控件加载文件的问题。
利用TreeView控件加载文件,必须遍历处所有的文件和文件夹. 深搜算法用到了递归. using System; using System.Collections.Generic; using Sy ...
- 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 ...
随机推荐
- Ubuntu 16.04出现:qmake: could not exec '/usr/lib/x86_64-linux-gnu/qt4/bin/qmake': No such file or directory
没有安装qt4-qmake,安装即可: sudo apt-get install qt4-qmake 参考: https://stackoverflow.com/questions/23703864/ ...
- 014 DNS
解析主机名 Router>en Password: Router#config t Enter configuration commands, one per line. End with C ...
- 混合使用Delphi和C ++(附下载)
您想将C ++添加到Delphi应用程序中吗?或者将Delphi代码添加到C ++应用程序中?这是如何做. 您可能不知道的一件事是如何在RAD Studio中集成C ++和Delphi语言.您可以将单 ...
- 远程查看日志-linux
ssh 连接服务器 ssh user@www.xxx.com -p60022 用户名@ip 端口 进入日志所在目录 cat FILENAME 查看文本文件,P.S. 在查较大文件时为了避免刷屏,请使用 ...
- APDU命令的结构和处理【转】
本文转载自:http://blog.csdn.net/yonghenzhita/article/details/36402525 版权声明:本文为博主原创文章,未经博主允许不得转载. 简单说,IFD( ...
- YTU 2520: 小慧唱卡拉OK
2520: 小慧唱卡拉OK 时间限制: 1 Sec 内存限制: 128 MB 提交: 478 解决: 207 题目描述 小慧唱歌非常好听,小鑫很喜欢听小慧唱歌,小鑫最近又想听小慧唱歌了,于是小鑫请 ...
- .net 接收存储过程的返回值 。。。。
.net 接收存储过程的返回值 .... Posted on 2009-06-10 20:26 且行且思 阅读(...) 评论(...) 编辑 收藏 例如在向数据库添加新数据时,需要检测是否有重复 本 ...
- python-----写入txt用法
代码如下: #!/usr/bin/env python # -*- coding: utf-8 -*- # @Time : 2019/1/14 11:23 # @Author : zxb file_p ...
- Spring通过注解注入有参
1.通过注解方式注入有参的构造函数 把@Autowired注解放在构造函数上方,在构造函数里写上需要注入的形参即可 2.通过XML配置文件方式定义有参构造函数
- Java DAO模式
DAO模式: DAO(DateAccessObject,数据存取对象) 位于业务逻辑和持久化数据之间,实现对持久化数据的访问. DAO组成: DAO接口: 数据库的所有操作定义成抽象方法,可以提供多种 ...