NOIP 2014【斗地主】
这真是道大火题。
因为保证数据随机,所以开始很多人直接用搜索 + 贪心水过去了,后来,为了遏制骗分这种不良风气的传播,各大 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【斗地主】的更多相关文章
- NOIP 2014 提高组 题解
NOIP 2014 提高组 题解 No 1. 生活大爆炸版石头剪刀布 http://www.luogu.org/problem/show?pid=1328 这是道大水题,我都在想怎么会有人错了,没算法 ...
- Luogu 2668 NOIP 2015 斗地主(搜索,动态规划)
Luogu 2668 NOIP 2015 斗地主(搜索,动态规划) Description 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来 ...
- Luogu 1351 NOIP 2014 联合权值(贪心,计数原理)
Luogu 1351 NOIP 2014 联合权值(贪心,计数原理) Description 无向连通图 G 有 n 个点,n-1 条边.点从 1 到 n 依次编号,编号为 i 的点的权值为 Wi, ...
- noip 2014 提高组初赛
noip 2014 提高组初赛 一. TCP协议属于哪一层协议( ) A. 应用层 B. 传输层 C. 网络层 D. 数据链路层 B TCP(传输控制协议) 若有变量int a; float: x, ...
- 洛谷P1328==codevs3716 生活大爆炸版石头剪刀布[NOIP 2014 day1 T1]
P1328 生活大爆炸版石头剪刀布 1.8K通过 2.6K提交 题目提供者2014白永忻 标签模拟NOIp提高组2014 难度普及- 提交该题 讨论 题解 记录 最新讨论 Who can help m ...
- noip 2014 总结
2014 年的noip 已经结束了,成绩从个人而言不是特别的理想,从今年题的逗的程度,本来是个xxx 就被玩脱了= = 当然现在后悔事没有用的了,不过第二天全屏技术的在最后一小时看到了两道题的错误,然 ...
- NOIP 2014 pj & tg
由于我太弱,去了pj组= = ============================== T1: 傻逼暴力 T2: 傻逼暴力+判断+更新 T3: 手画一下就知道了.算出这个点在第几圈,再使劲yy下在 ...
- [NOIP 2014复习]第三章:动态规划——NOIP历届真题回想
背包型动态规划 1.Wikioi 1047 邮票面值设计 题目描写叙述 Description 给定一个信封,最多仅仅同意粘贴N张邮票,计算在给定K(N+K≤40)种邮票的情况下(假定全部的邮票数量都 ...
- 基础算法(搜索):NOIP 2015 斗地主
Description 牛牛最近迷上了一种叫斗地主的扑克游戏.斗地主是一种使用黑桃.红心.梅花.方片的A到K加上大小王的共54张牌来进行的扑克牌游戏.在斗地主中,牌的大小关系根据牌的数码表示如下:3& ...
随机推荐
- HttpClient 模拟登录搜狐微博
http://mengyang.iteye.com/blog/575671 第一次遇到一个这样的问题,"PKIX path building failed" 异常 详解异常: ...
- BCG菜单button的简单使用
一,新建一个BCGprojectCBCGPMenuButton,基于对话框. 二.添加一个button,并关联一个CButton类型的变量m_btn1.然后手动将类型改CBCGPMenuButton成 ...
- containsKey使用方法
作用是判断Map中是否有所需要的键值,下面是具体的代码: public static void main(String[] args) { Map<String, String> map ...
- hdu1316
链接:pid=1316" target="_blank">点击打开链接 题意:问区间[a,b]中有多少斐波那契数 代码: #include <iostream ...
- 软件工程第3次作业——Visual Studio 2017下针对代码覆盖率的C/C++单元测试
本项目Github地址(同时包括两个作业项目): Assignment03 -- https://github.com/Oberon-Zheng/SoftwareEngineeringAssignme ...
- mysql 查排名
SET @amount=0; SET @rank=1; SET @shunxu=0; SELECT tmp2.id AS id,tmp2.name AS NAME,tmp2.amount AS ...
- HDU 2112 HDU Today(STL MAP + Djistra)
题目链接:HDU Today 立即集训要開始,抓紧时间练练手,最短路的基础题,第一次用STL的map 题目非常水,可是错了N遍.手贱了.本题不优点理的就是把地名转化为数字 #include <i ...
- 卡特兰数-Catalan数
卡特兰数的含义: 说到卡特兰数,就不得不提及卡特兰数序列.卡特兰数序列是一个整数序列.其通项公式是我们从中取出的就叫做第n个卡特兰数数,前几个卡特兰数数是:1, 1, 2, 5, 14, 42, 13 ...
- Chapter 4 马尔科夫链
4.1 引言 现在要研究的是这样一种过程: 表示在时刻的值(或者状态),想对一串连续时刻的值,比如:,, ... 建立一个概率模型. 最简单的模型就是:假设都是独立的随机变量,但是通常这种假设都是没什 ...
- C#与Java在修饰符上的不同
1.readonly 修饰符仅用于修饰类的数据成员.正如其名字说的,一旦它们已经进行了写操作.直接初始化或在构造函数中对其进行了赋值,数据成员就只能对其进行读取. readonly 和 const 数 ...