题目描述

牛牛最近迷上了一种叫斗地主的扑克游戏。斗地主是一种使用黑桃、红心、梅花、方片的A到K加上大小王的共54张牌来进行的扑克牌游戏。在斗地主中,牌的大小关系根据牌的数码表示如下:3<4<5<6<7<8<9<10<J<Q<K<A<2<小王<大王,而花色并不对牌的大小产生影响。每一局游戏中,一副手牌由n张牌组成。游戏者每次可以根据规定的牌型进行出牌,首先打光自己的手牌一方取得游戏的胜利。现在,牛牛只想知道,对于自己的若干组手牌,分别最少需要多少次出牌可以将它们打光。请你帮他解决这个问题。需要注意的是,本题中游戏者每次可以出手的牌型与一般的斗地主相似而略有不同。具体规则如下:

输入

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

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

输出

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

样例输入1

1 8

7 4

8 4

9 1

10 4

11 1

5 1

1 4

1 1

样例输出1

3

样例说明:共有1组手牌,包含8张牌:方片7,方片8,黑桃9,方片10,黑桃J,黑桃5,方片A以及黑桃A。可以通过打单顺子(方片7,方片8,黑桃9,方片10,黑桃J),单张牌(黑桃5)以及对子牌(黑桃A以及方片A)在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

数据范围

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

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

时空限制

1024M,2s

【题解】

一道喜闻乐见的大模拟大搜索题,看到了之后就一直跃跃欲试想做一做。考试主要的时间都用来打这道题,刚开始是bfs,卡到直接死机。后来改成dfs,成功过了样例,水过6个点。总觉得dfs不如bfs适合找最优解,但是其实dfs也可以通过剪枝遏制不必要的搜索,而像这种情况非常多的题目bfs要尝试每一种情况的下一步,消耗过大反而不适宜。

正解也是暴搜,但是搜得比我有技巧多了。尽量先打出较多牌来剪枝,最后那些没有被各种特效打出去的牌就直接一视同仁st++就好了。在考试代码上加了最后这条优化瞬间从TLE变成AC,其实想想打单种牌这么简单的事,一句话就解决了何必再来一层dfs呢~据说这道题体现了用各种条件层层剪枝的重要性,但是对于我来说,还是对所谓的“高明的处理方法”更有体会吧。

#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
int t,n,p[],a1,a2,jg;
bool qk;
void init()
{
memset(p,,sizeof(p));
for(int i=;i<=n;i++)
{
scanf("%d%d",&a1,&a2);
if(a1==) p[]++;
if(a1==) p[]++;
if(a1>) p[a1-]++;
if(!a1)
{
if(a2==) p[]++;
if(a2==) p[]++;
}
}
}
void dfs(int st)
{
if(st>=jg) return;
qk=;
int ss,sz,sf;
ss=sz=sf=;
for(int i=;i<=;i++)
{
if(p[i]) qk=;
if(p[i]>=) sz++;
if(p[i]>=) ss++;
if(p[i]>=) sf++;
}
if(qk)
{
jg=st;
return;
}
if(st==jg-) return; int temp;
temp=;
for(int i=temp+;i<=;i++)
if(p[i])
{
temp=i;
for(int j=i+;j<=;j++)
{
if(p[j]) temp=j;
else break;
}
if(temp>=i+)
{
for(int j=i;j<=temp;j++) p[j]--;
st++;
dfs(st);
for(int j=i;j<=temp;j++) p[j]++;
st--;
}
temp++;
}
if(sz>=)
{
temp=;
for(int i=temp+;i<=;i++)
if(p[i]>=)
{
temp=i;
for(int j=i+;j<=;j++)
{
if(p[j]>=) temp=j;
else break;
}
if(temp>=i+)
{
for(int j=i;j<=temp;j++) p[j]-=;
st++;
dfs(st);
for(int j=i;j<=temp;j++) p[j]+=;
st--;
}
if(temp-i+==sz) break;
temp++;
}
}
if(ss>=)
{
temp=;
for(int i=temp+;i<=;i++)
if(p[i]>=)
{
temp=i;
for(int j=i+;j<=;j++)
{
if(p[j]>=) temp=j;
else break;
}
if(temp>=i+)
{
for(int j=i;j<=temp;j++) p[j]-=;
st++;
dfs(st);
for(int j=i;j<=temp;j++) p[j]+=;
st--;
}
if(temp-i+==ss) break;
temp++;
}
} if(sf)
for(int i=;i<=;i++)
if(p[i]>=)
{
p[i]-=,st++;
for(int j=;j<=;j++)
if(p[j])
{
p[j]--;
for(int jk=j;jk<=;jk++)
if(p[jk])
{
p[jk]--;
dfs(st);
p[jk]++;
}
p[j]++;
}
if(sz>=)
{
for(int j=;j<=;j++)
if(p[j]>=)
{
p[j]-=;
for(int jk=j;jk<=;jk++)
if(p[jk]>=)
{
p[jk]-=;
dfs(st);
p[jk]+=;
}
p[j]+=;
}
}
p[i]+=;
st--;
if(sf==) break;
} if(ss)
for(int i=;i<=;i++)
if(p[i]>=)
{
p[i]-=,st++;
for(int j=;j<=;j++)
if(p[j]&&j!=i)
{
p[j]--;
dfs(st);
p[j]++;
}
if(sz)
for(int j=;j<=;j++)
if(p[j]>=)
{
p[j]-=;
dfs(st);
p[j]+=;
}
p[i]+=,st--;
if(ss==) break;
} if(p[]&&p[])
{
p[]--,p[]--,st++;
dfs(st);
p[]++,p[]++,st--;
} for(int i=;i<=;i++)
if(p[i])
st++;
if(st<jg) jg=st;
}
int main()
{
scanf("%d%d",&t,&n);
for(int l=;l<=t;l++)
{
init();
jg=0x3fff;
dfs();
printf("%d\n",jg);
}
return ;
}

landlords

斗地主[NOIP2015]的更多相关文章

  1. 斗地主 (NOIP2015 Day1 T3)

    斗地主 张牌,因为它可以连在K后, 总体思路为 先出炸弹和四带二 再出三带一 再把对牌和单牌出完 记录并更新Answer,后枚举顺子,并继续向下搜索. 注意:弄明白题意,题目描述不太清楚....另外, ...

  2. 洛谷P2668 斗地主 [NOIP2015]

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

  3. NOIP2015斗地主[DFS 贪心]

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

  4. BZOJ 4325: NOIP2015 斗地主

    4325: NOIP2015 斗地主 Time Limit: 30 Sec  Memory Limit: 1024 MBSubmit: 684  Solved: 456[Submit][Status] ...

  5. NOIP2015 斗地主(搜索+剪枝)

    4325: NOIP2015 斗地主 Time Limit: 30 Sec  Memory Limit: 1024 MBSubmit: 270  Solved: 192[Submit][Status] ...

  6. [补档][NOIP2015] 斗地主

    [NOIP2015] 斗地主 题目 传送门:http://cogs.pro/cogs/problem/problem.php?pid=2106 INPUT 第一行包含用空格隔开的2个正整数Tn,表示手 ...

  7. LOJ2422 NOIP2015 斗地主 【搜索+贪心】*

    LOJ2422 NOIP2015 斗地主 LINK 题目大意很简单,就是问你斗地主的一分手牌最少多少次出完 然后我们发现对于一种手牌状态,不考虑顺子的情况是可以贪心做掉的 然后我们直接枚举一下顺子出牌 ...

  8. 【BZOJ4325】NOIP2015 斗地主 搜索+剪枝

    [BZOJ4325]NOIP2015 斗地主 Description 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游戏.在斗 ...

  9. 2106. [NOIP2015] 斗地主

        2106. [NOIP2015] 斗地主 ★★★☆   输入文件:landlords.in   输出文件:landlords.out   简单对比 时间限制:2 s   内存限制:1025 M ...

随机推荐

  1. RabbitMQ~开篇与环境部署

    想写这篇文章很久了,今天终于有时间总结一下,一个大型的系统里,消息中间件是必不可少的,它将并发环境处理的数据异步进行处理,有效的提高了系统的并发能力,有很多系统的瓶颈点都在于此,而消息中间件在这个时候 ...

  2. Builder模式详解及其在Android开发中的应用

    一.引言 在Android开发中,采用Builder模式的代码随处可见,比如说Android系统对话框AlertDialog的使用或者是Android中的通知栏(Notification)的使用,又比 ...

  3. 简单理解js闭包

    什么是闭包?我们先来看一段代码: function a() { var n = 0; function inc() { n++; console.log(n); } inc(); inc(); } a ...

  4. MVC过滤器之添加LoginAttribute,浏览器bug:重定向次数太多

    以前在写登录Action过滤时,都在每个Controller前写上CheckLoginAttribute:这次决定偷懒试一下能否将所有Action和Controller统一过滤: 出bug的代码是这样 ...

  5. html模板中的数字保留两位小数

    <script> //html模板中的数字保留两位小数 function formatCurrency(num) { num = num.toString().replace(/\$|\, ...

  6. MyBatis-sql映射文件

    Sql映射文件 MyBatis真正的力量是在映射语句中.这里是奇迹发生的地方.对于所有的力量,SQL映射的XML文件是相当的简单.当然如果你将它们和对等功能的JDBC代码来比较,你会发现映射文件节省了 ...

  7. Python爬虫小白---(二)爬虫基础--Selenium PhantomJS

    一.前言   前段时间尝试爬取了网易云音乐的歌曲,这次打算爬取QQ音乐的歌曲信息.网易云音乐歌曲列表是通过iframe展示的,可以借助Selenium获取到iframe的页面元素, 而QQ音乐采用的是 ...

  8. EJB系列 - EJB高级概念

    本人博客文章网址:https://www.peretang.com/ejb-advanced-concepts/ EJB内幕 幕后的EJB:容器会为每一个bean实例自动生成称为EJB对象的代理, 由 ...

  9. Python 3从入门到精通01-环境搭建

    本系列开始介绍Python3的基础教程,为什么要选中Python 3呢?因为最近看到一些资料和课程,都是Python 3授课的,例如,大数据,机器学习,数据挖掘等等:还有一个目的,我想彻底地,系统地学 ...

  10. Java数据类型在实际开发中的应用二枚举类型

    在实际编程中,往往存在着这样的"数据集",它们的数值在程序中是稳定的,而且"数据集"中的元素是有限的.在JDK1.5之前,人们用接口来描述这一种数据类型. 1. ...