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 ...
随机推荐
- Use Elasticksearch to solve TOP N issue
The raw data is like timestamp, router, interface, src_ip, dst_ip, protocol, pkts 10000000, 1.1.1.1 ...
- spring-boot上传文件MultiPartFile获取不到文件问题解决
1.现象是在spring-boot里加入commons-fileupload jar并且配置了mutilPart的bean,在upload的POST请求后,发现 multipartRequest.ge ...
- 解决confluence的乱码问题
使用confluence时发现一些含有中文的页面中,中文都变成了问号. 继续搜索解决方案,发现时数据库中数据的格式不对, 在mysql中输入以下命令: mysql> show variabl ...
- 学习swift从青铜到王者之字符串和运算符02
1 字符和字符串初步 var c :Character = "a" 2 构造字符串 let str1 = "hello" let str2 = " ...
- 【Nginx】负载均衡-加权轮询策略剖析
转自:江南烟雨 本文介绍的是客户端请求在多个后端服务器之间的均衡,注意与客户端请求在多个nginx进程之间的均衡相区别. 如果Nginx是以反向代理的形式配置运行,那么对请求的实际处理需要转发到后端服 ...
- CSS (二)解析CSS盒子
话说.一写博客还有些莫名的兴奋感-- 这几天一直挤时间忙于赶牛腩视频,迟到的CSS盒子.请谅解. CSS盒子,一開始听起来还有点高大上的赶脚. 后来了解之后,发现事实上非常easy理解.从功能上讲非常 ...
- AutoCAD如何方便截图放到Word文档,改成白底黑字
将模型视图切换到布局2即可 比如下图所示的效果 先回到模型视图把所有线条颜色都改成白色,然后添加适当的标注(比如要受力分析,则在CAD中绘制箭头也很方便的),文字说明.然后切换到布局2就OK ...
- C++虚函数表剖析
关键词:虚函数.虚表,虚表指针,动态绑定,多态 一.概述 为了实现C++的多态,C++使用了一种动态绑定的技术. 这个技术的核心是虚函数表(下文简称虚表).本文介绍虚函数表是怎样实现动态绑定的. 二. ...
- Maven+Mybatis+Spring+SpringMVC实现分页查询(附源代码)
以下小宝鸽将分享一篇Mybatis分页功能的博文,以下将给出具体的步骤.跟着博主的节奏肯定能实现.另外最后还会附上整个project的源代码.假设是没有使用过maven的猿友可自行下载相关的jar包就 ...
- 【bzoj3676】[Apio2014]回文串 —— 回文自动机的学习
写题遇上一棘手的题,[Apio2014]回文串,一眼看过后缀数组+Manacher.然后就码码码...过是过了,然后看一下[Status],怎么慢这么多,不服..然后就搜了一下,发现一种新东西——回文 ...