【BZOJ4325】【NOIP2015】斗地主 搜索
题目描述
就是给你一副牌,问你最少几次能出完。
详细规则见规则
\(n\leq 23\)
题解
NOIP的数据非常水,错误一大堆的程序都能AC。
因为顺子对答案的影响最大,所以先枚举顺子进行搜索。
接下来网上很多的程序都直接计算答案了。实际上还要进行一次搜索,枚举拆牌的方案,有以下几种:
1.一组四张牌可以拆成两组两张牌
2.一组四张牌可以拆成一组一张牌+一组三张牌
3.一组三张牌可以拆成一组一张牌+一组两张牌
4.一组两张牌可以拆成两组一张牌
5.一对王可以拆成两组一张牌
然后就可以计算方案。
加个最优性剪枝就可以过了。
时间复杂度:\(O(???)\)
代码
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cstdlib>
#include<ctime>
#include<utility>
using namespace std;
typedef long long ll;
typedef unsigned long long ull;
typedef pair<int,int> pii;
int s[20];
int a[20],c[20];
int ans,n;
int calc2()
{
int res=0;
memcpy(c,a,sizeof a);
while(c[4]&&c[2]>=2)
{
c[4]--;
c[2]-=2;
res++;
}
while(c[4]&&c[1]>=2)
{
c[4]--;
c[1]-=2;
res++;
}
while(c[3]&&c[2])
{
c[3]--;
c[2]--;
res++;
}
while(c[3]&&c[1])
{
c[3]--;
c[1]--;
res++;
}
return res+c[1]+c[2]+c[3]+c[4];
}
int calc()
{
int ans=calc2();
if(a[4])
{
a[4]--;
a[2]+=2;
ans=min(ans,calc());
a[4]++;
a[2]-=2;
}
if(a[4])
{
a[4]--;
a[3]++;
a[1]++;
ans=min(ans,calc());
a[4]++;
a[3]--;
a[1]--;
}
if(a[3])
{
a[3]--;
a[1]++;
a[2]++;
ans=min(ans,calc());
a[1]--;
a[2]--;
a[3]++;
}
if(a[2])
{
a[2]--;
a[1]+=2;
ans=min(ans,calc());
a[1]-=2;
a[2]++;
}
return ans;
}
int calc3()
{
int i;
memset(a,0,sizeof a);
int res=0;
if(s[0]>=2)
res++;
else if(s[0]==1)
a[1]++;
for(i=1;i<=13;i++)
a[s[i]]++;
res+=calc();
memset(a,0,sizeof a);
for(i=1;i<=13;i++)
a[s[i]]++;
a[1]+=s[0];
return min(res,calc());
}
void dfs(int now)
{
if(now>=ans)
return;
int v=calc3();
ans=min(ans,now+v);
int i,j,k;
for(i=2;i<=13;i++)
{
j=i;
while(j<=13&&s[j]>=3)
{
j++;
if(j-i>=2)
{
for(k=i;k<=j-1;k++)
s[k]-=3;
dfs(now+1);
for(k=i;k<=j-1;k++)
s[k]+=3;
}
}
}
for(i=2;i<=13;i++)
{
j=i;
while(j<=13&&s[j]>=2)
{
j++;
if(j-i>=3)
{
for(k=i;k<=j-1;k++)
s[k]-=2;
dfs(now+1);
for(k=i;k<=j-1;k++)
s[k]+=2;
}
}
}
for(i=2;i<=13;i++)
{
j=i;
while(j<=13&&s[j]>=1)
{
j++;
if(j-i>=5)
{
for(k=i;k<=j-1;k++)
s[k]-=1;
dfs(now+1);
for(k=i;k<=j-1;k++)
s[k]+=1;
}
}
}
for(i=2;i<=13;i++)
if(s[i]==4)
{
s[i]-=4;
for(j=i+1;j<=13;j++)
if(s[j]==4)
{
s[j]-=4;
dfs(now+2);
s[j]+=4;
}
s[i]+=4;
}
}
void solve()
{
int x,y;
memset(s,0,sizeof s);
int i;
for(i=1;i<=n;i++)
{
scanf("%d%d",&x,&y);
if(x==1)
x=13;
else if(x)
x--;
s[x]++;
}
ans=0x7fffffff;
dfs(0);
printf("%d\n",ans);
}
int main()
{
int t;
scanf("%d%d",&t,&n);
while(t--)
solve();
return 0;
}
【BZOJ4325】【NOIP2015】斗地主 搜索的更多相关文章
- 【BZOJ4325】NOIP2015 斗地主 搜索+剪枝
[BZOJ4325]NOIP2015 斗地主 Description 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游戏.在斗 ...
- 2018.11.01 bzoj4325: NOIP2015 斗地主(贪心+搜索)
传送门 原来一直以为是一道大模拟. 没想到是一道搜索+最优性剪枝 如何搜最优呢? 我们考虑怎么最快出完. 大概是应该尽量出当前能出出去最多的吧. 于是我们选择优先出顺子. 这样做有什么好处呢? 我们会 ...
- 【BZOJ4325】NOIP2015 斗地主 搜索+贪心
这个东西考试的时候一眼以为状压就压炸了考试又了一下午.....最后我打出来发现后几个点10min都过不去,我大概算了一下,可能是吧.......最后一脸懵逼的我去怂了正解,我们发现只要确定了顺子就可以 ...
- bzoj4325: NOIP2015 斗地主(爆搜+模拟)
去年的我还不会打斗地主呵呵 觉得这道题挺难的..抄了一遍题解,感触挺多的= = 首先出牌的方式太多了不能每次都枚举所有的出牌方式, 于是分成两部分:1.顺子 2.带牌等其他 每次dfs都搜顺子,而且顺 ...
- NOIP2015 斗地主(搜索+剪枝)
4325: NOIP2015 斗地主 Time Limit: 30 Sec Memory Limit: 1024 MBSubmit: 270 Solved: 192[Submit][Status] ...
- LOJ2422 NOIP2015 斗地主 【搜索+贪心】*
LOJ2422 NOIP2015 斗地主 LINK 题目大意很简单,就是问你斗地主的一分手牌最少多少次出完 然后我们发现对于一种手牌状态,不考虑顺子的情况是可以贪心做掉的 然后我们直接枚举一下顺子出牌 ...
- NOIP2015斗地主[DFS 贪心]
题目描述 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游戏.在斗地主中,牌的大小关系根据牌的数码表示如下:3<4< ...
- BZOJ 4325: NOIP2015 斗地主
4325: NOIP2015 斗地主 Time Limit: 30 Sec Memory Limit: 1024 MBSubmit: 684 Solved: 456[Submit][Status] ...
- 2106. [NOIP2015] 斗地主
2106. [NOIP2015] 斗地主 ★★★☆ 输入文件:landlords.in 输出文件:landlords.out 简单对比 时间限制:2 s 内存限制:1025 M ...
- NOIP2015斗地主题解 7.30考试
问题 B: NOIP2015 斗地主 时间限制: 3 Sec 内存限制: 1024 MB 题目描述 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共 ...
随机推荐
- easyui datagrid 相关取数据总结
easyui 中datagrid$('#dg').datagrid('getSelected');返回第一个被选中的行或如果没有选中的行则返回null.$('#dg').datagrid('getSe ...
- 详解ES5和ES6的继承
ES5继承 构造函数.原型和实例的关系:每一个构造函数都有一个原型对象,每一个原型对象都有一个指向构造函数的指针,而每一个实例都包含一个指向原型对象的内部指针, 原型链实现继承 基本思想:利用原型让一 ...
- c++入门之结构体初步
结构体实际上是一种数据结构的雏形,对结构体的灵活使用很多时候可以带来很多便利.下面给出一个关于结构体的程序: #include "iostream" # include " ...
- shell脚本--编写CGI代码(shell结合html)以及环境变量
实现shell和html标签混合的方式编写代码: 推荐 初始CGI ,看完大概之后,大概对cgi有个大体的印象.下面是编写混合代码的示例: #!/bin/bash #index.cgi echo & ...
- CMD管道命令使用
Windows netstat 查看端口.进程占用 开始--运行--cmd 进入命令提示符 输入netstat -ano 即可看到所有连接的PID 之后在任务管理器中找到这个PID所对应的程序如果任务 ...
- oracle创建视图时一些问题
这几天创建视图的时候,遇见的问题. 一:创建视图的时候Oracle-报错:文字与格式字符串不匹配(ORA-01861) 我创建的时候用的 是to_date 然后我改成了to_char select X ...
- opencv2\core\cuda.hpp(106): error C2059: 语法错误:“常量”
在 cuda.hpp 中, virtual void free(GpuMat* mat) = 0; -> virtual void _free(GpuMat* mat) = 0;
- hdu1421_搬寝室
题目:搬寝室 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1421 #include<stdio.h> #include<algor ...
- 对C#调用C++的dll的一点思考
最近在对接C++程序的时候碰到了一些问题,然后花了一段时间才解决,今天就这些小问题来做一个总结,很多时候由于对另外一种开发语言的不熟悉,会在使用的过程中遇到很多的问题,这些问题看似简单但是背后却有很多 ...
- saltstack二
配置管理 haproxy的安装部署 haproxy各版本安装包下载路径https://www.haproxy.org/download/1.6/src/,跳转地址为http,改为https即可 创建相 ...