这真是道大火题。

因为保证数据随机,所以开始很多人直接用搜索 + 贪心水过去了,后来,为了遏制骗分这种不良风气的传播,各大 OJ 相继推出了斗地主加强版……

正解:

先爆搜顺子,枚举打或不打,打多少张。对于剩下的散牌做 DP,最少需要多少次打完。

设 \(f[i][j][k][l]\) 表示四张牌的剩 \(i\) 张,三张牌的剩 \(j\) 张,两张牌的剩 \(k\) 张,一张牌的剩 \(l\) 张,最少需要多少次打完。因为不考虑打顺子,所以点数与打完的次数无关。然后按照题目除顺子之外的打法转移即可。

与加强版的区别就是网能不能看成普通对子。

注意

记忆化搜索的值莫名其妙的变化了多半是数组越界。

搜索不要没想清楚就随便剪枝,可能不知不觉中就剪挂了。

代码是加强版的。

#include <bits/stdc++.h>
using namespace std; #define N 25
#define RG register inline int gi()
{
RG int ret; RG char ch;
ret=0, ch=getchar();
while (ch < '0' || ch > '9')
ch=getchar();
while (ch >= '0' && ch <= '9')
ret=(ret<<3)+(ret<<1)+ch-'0', ch=getchar();
return ret;
} int tong[N],f[N][N][N][N];
int ans,n; inline int cal(int i,int j,int k,int l)
{
if (f[i][j][k][l] || (!i && !j && !k && !l))
return f[i][j][k][l];
RG int ret;
ret=i+j+k+l;
if (i)
ret=min(ret,min(cal(i-1,j+1,k,l+1),cal(i-1,j,k+2,l)));
if (j)
ret=min(ret,cal(i,j-1,k+1,l+1));
if (k)
ret=min(ret,cal(i,j,k-1,l+2));
if (i && k > 1)
ret=min(ret,cal(i-1,j,k-2,l)+1);
if (i && l > 1)
ret=min(ret,cal(i-1,j,k,l-2)+1);
if (j && k)
ret=min(ret,cal(i,j-1,k-1,l)+1);
if (j && l)
ret=min(ret,cal(i,j-1,k,l-1)+1);
return f[i][j][k][l]=ret;
} inline void dfs(int dep)
{
if (dep >= ans)
return;
RG int i,j,k,l,tot;
for (i=3; i<11; ++i)
{
if (!tong[i])
continue;
tot=1, j=i+1;
while (tong[j])
tot++, j++;
if (tot < 5)
continue;
for (j=5; j<=tot; ++j)
{
for (k=0; k<j; ++k)
tong[i+k]--;
dfs(dep+1);
for (k=0; k<j; ++k)
tong[i+k]++;
}
}
for (i=3; i<13; ++i)
{
if (tong[i] < 2)
continue;
tot=1, j=i+1;
while (tong[j] >= 2)
tot++, j++;
if (tot < 3)
continue;
for (j=3; j<=tot; ++j)
{
for (k=0; k<j; ++k)
tong[i+k]-=2;
dfs(dep+1);
for (k=0; k<j; ++k)
tong[i+k]+=2;
}
}
for (i=3; i<14; ++i)
{
if (tong[i] < 3)
continue;
tot=1, j=i+1;
while (tong[j] >= 3)
tot++, j++;
if (tot < 2)
continue;
for (j=2; j<=tot; ++j)
{
for (k=0; k<j; ++k)
tong[i+k]-=3;
dfs(dep+1);
for (k=0; k<j; ++k)
tong[i+k]+=3;
}
}
i=j=k=l=0;
for (tot=2; tot<15; ++tot)
if (!tong[tot])
continue;
else if (tong[tot] == 1)
l++;
else if (tong[tot] == 2)
k++;
else if (tong[tot] == 3)
j++;
else if (tong[tot] == 4)
i++;
if (tong[0] == 2)
ans=min(ans,dep+cal(i,j,k,l)+1);
l+=tong[0];
// if (tong[0] == 2)
// k++;
// else if (tong[0] == 1)
// l++;
ans=min(ans,dep+cal(i,j,k,l));
} inline void work()
{
RG int i,x;
for (i=1; i<=n; ++i)
x=gi(), tong[x]++, x=gi();
tong[14]=tong[1];
ans=n;
dfs(0);
printf("%d\n",ans);
for (i=0; i<15; ++i)
tong[i]=0;
} int main()
{
RG int t;
t=gi(), n=gi();
while (t--)
work();
return 0;
}

NOIP 2014【斗地主】的更多相关文章

  1. NOIP 2014 提高组 题解

    NOIP 2014 提高组 题解 No 1. 生活大爆炸版石头剪刀布 http://www.luogu.org/problem/show?pid=1328 这是道大水题,我都在想怎么会有人错了,没算法 ...

  2. Luogu 2668 NOIP 2015 斗地主(搜索,动态规划)

    Luogu 2668 NOIP 2015 斗地主(搜索,动态规划) Description 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来 ...

  3. Luogu 1351 NOIP 2014 联合权值(贪心,计数原理)

    Luogu 1351 NOIP 2014 联合权值(贪心,计数原理) Description 无向连通图 G 有 n 个点,n-1 条边.点从 1 到 n 依次编号,编号为 i 的点的权值为 Wi, ...

  4. noip 2014 提高组初赛

    noip 2014 提高组初赛 一. TCP协议属于哪一层协议( ) A. 应用层 B. 传输层 C. 网络层 D. 数据链路层 B TCP(传输控制协议) 若有变量int a; float: x, ...

  5. 洛谷P1328==codevs3716 生活大爆炸版石头剪刀布[NOIP 2014 day1 T1]

    P1328 生活大爆炸版石头剪刀布 1.8K通过 2.6K提交 题目提供者2014白永忻 标签模拟NOIp提高组2014 难度普及- 提交该题 讨论 题解 记录 最新讨论 Who can help m ...

  6. noip 2014 总结

    2014 年的noip 已经结束了,成绩从个人而言不是特别的理想,从今年题的逗的程度,本来是个xxx 就被玩脱了= = 当然现在后悔事没有用的了,不过第二天全屏技术的在最后一小时看到了两道题的错误,然 ...

  7. NOIP 2014 pj & tg

    由于我太弱,去了pj组= = ============================== T1: 傻逼暴力 T2: 傻逼暴力+判断+更新 T3: 手画一下就知道了.算出这个点在第几圈,再使劲yy下在 ...

  8. [NOIP 2014复习]第三章:动态规划——NOIP历届真题回想

    背包型动态规划 1.Wikioi 1047 邮票面值设计 题目描写叙述 Description 给定一个信封,最多仅仅同意粘贴N张邮票,计算在给定K(N+K≤40)种邮票的情况下(假定全部的邮票数量都 ...

  9. 基础算法(搜索):NOIP 2015 斗地主

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

随机推荐

  1. 聚合数据 iOS 项目开发实战:条码查询器

    记录下,聚合数据 iOS 项目开发实战:条码查询器:视频地址:http://www.jikexueyuan.com/course/324.html 条码查询API:https://www.juhe.c ...

  2. hdfs笔记

    Distributed File System 数据量越来越多,在一个操作系统管辖的范围存不下了,那么就分配到更多的操作系统管理的磁盘中,但是不方便管理和维护,因此迫切需要一种系统来管理多台机器上的文 ...

  3. 转:DDR中端接技术基本概念

    DDR中端接技术基本概念  版权声明:转载请注明出处:http://blog.csdn.net/lg2lh https://blog.csdn.net/lg2lh/article/details/90 ...

  4. cocos2dx中使用iconv转码(win32,iOS,Android)

    首先贴下环境:Win7 64, NDK r8e, libiconv-1.14, cygwin 一 Win32环境配置 Cocos2D-X自带有win32上的iconv库.仅仅须要配置一下就可以使用. ...

  5. 开发ActiveX控件调用另一个ActiveX系列2——调试ActiveX

    关于调试ActiveX控件,有若干方法,例如可以建一个MFC工程调用调试,我则倾向于使用附加到浏览器进程,因为浏览器才是真正运行的环境. 打开加载ActiveX的目标页面,当然希望我们的调试内容不是自 ...

  6. scrapy递归抓取网页数据

    scrapy spider的parse方法能够返回两种值:BaseItem.或者Request.通过Request能够实现递归抓取. 假设要抓取的数据在当前页,能够直接解析返回item(代码中带**凝 ...

  7. GitHub使用问题(遇到一个记一个)

    1.如何创建文件夹: 如图,Create new files,点击后,若需要创建文件,输入文件名即可,但如果创建的是文件夹,需要在文件夹名后 加一个 '/'斜杠,然后就变成文件夹了

  8. Redis专题(2):Redis数据结构底层探秘

    前言 上篇文章Redis闲谈(1):构建知识图谱介绍了redis的基本概念.优缺点以及它的内存淘汰机制,相信大家对redis有了初步的认识.互联网的很多应用场景都有着Redis的身影,它能做的事情远远 ...

  9. PLSQL怎样导出oracle表结构

    tools->export tables 是导出表结构还有数据 tools->export user objects是导出表结构   可以用tools->export tables ...

  10. SQL Server 存储过程的几种常见写法分析,我们该用那种写法

    本文出处: http://www.cnblogs.com/wy123/p/5958047.html 最近发现还有不少做开发的小伙伴,在写存储过程的时候,在参考已有的不同的写法时,往往很迷茫,不知道各种 ...