题目描述

牛牛最近迷上了一种叫斗地主的扑克游戏。斗地主是一种使用黑桃、红心、梅花、方片的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. javascript对象(1)

    今天说面向对象,嗯,不是那个对象,是这个对象. 接下来就开始今天的内容: 什么是面向对象: 就是把数据及数据的操作方法放在一起,作为一个相互依存的整体----对象.对同类对象抽象出其共性,形成类. 类 ...

  2. 【Android Developers Training】 88. 使用备份API

    注:本文翻译自Google官方的Android Developers Training文档,译者技术一般,由于喜爱安卓而产生了翻译的念头,纯属个人兴趣爱好. 原文链接:http://developer ...

  3. 短信发送接口被恶意访问的网络攻击事件(四)完结篇--搭建WAF清理战场

    前言 短信发送接口被恶意访问的网络攻击事件(一)紧张的遭遇战险胜 短信发送接口被恶意访问的网络攻击事件(二)肉搏战-阻止恶意请求 短信发送接口被恶意访问的网络攻击事件(三)定位恶意IP的日志分析脚本 ...

  4. KVO底层实现原理,仿写KVO

    这篇文章简单介绍苹果的KVO底层是怎么实现的,自己仿照KVO的底层实现,写一个自己的KVO监听 #pragma mark--KVO底层实现 第一步:新建一个Person类继承NSObject Pers ...

  5. Unity 游戏框架搭建 (一) 概述

      为了重构手头的一款项目,翻出来当时未接触Unity时候收藏的视频<Unity项目架构设计与开发管理>,对于我这种初学者来说全是干货.简单的总结了一下,以后慢慢提炼. 关于Unity的架 ...

  6. XML 新手入门基础知识(复制,留着自己看)

    如果您是 XML 新手,本文将为您介绍 XML 文档的基础结构,以及创建构造良好的 XML 需要遵循的规则,包括命名约定.正确的标记嵌套.属性规则.声明和实体.您还可以从本文了解到 DTD 和 sch ...

  7. cygwin安装gcc/g++

    安装cygwin如果按照默认的方式一直点下去的话,安装完了会发现没有安装gcc/g++. 这个时候可以在安装文件的目录打开命令行,并输入: setup-x86_64.exe -q -P wget -P ...

  8. man ssh翻译(ssh命令中文手册)

    本文为命令ssh的man文档翻译,翻译了90%的内容,剩余是一些没必要翻译的东西,请见谅. 如此文有所疑惑,希望我的另一篇文章能解惑: SSH(1)                    BSD Ge ...

  9. [图形学] Chp14 GLU曲面裁剪函数程序示例及样条表示遗留问题

    样条表示这章已经看完,最后的GLU曲面裁剪函数,打算按书中的示例实现一下,其中遇到了几个问题. 先介绍一下GLU曲面裁剪函数的使用方法. 1 裁剪函数是成对出现的: gluBeginTrim和gluE ...

  10. Java ee el表达式

    以前在开发的时候,偶尔会遇到jsp页面不支持el表达式的情况. 这个的原因是因为El功能被关闭了, 当时的解决办法是关闭忽略.isELIgnored 设设置 但是为什么有时候不用设置也可以了呢.发现原 ...