[luogu 5301][bzoj 5503] [GXOI/GZOI2019] 宝牌一大堆
题面
好像ZJOI也考了一道麻将, 这是要发扬中华民族的赌博传统吗???
暴搜都不会打, 看到题目就自闭了, 考完出来之后看题解, \(dp\), 可惜自己想不出来...
对于国士无双(脑子中闪过了韩信)和七对子进行特判, 国士无双枚举哪张牌选两张即可, 七对子的话对于每种牌, 如果该种牌可以凑对子, 就将它凑对子对答案的贡献算出来, 排序后贪心地选七个最大的即可
国士无双
for(int i = 1; i <= 13; i++)
{
long long tmp = 1;
for(int j = 1; j <= 13; j++)
{
if(i == j)
{
if(a[gs[j]] < 2) tmp = 0;
else tmp *= c[a[gs[j]]][2] * (b[gs[j]] ? 4 : 1);
}
else
{
if(a[gs[j]] < 1) tmp = 0;
else tmp *= c[a[gs[j]]][1] * (b[gs[j]] ? 2 : 1);
}
}
ans = max(ans, tmp * 13);
}
七对子
cnt = 0;
for(int i = 1; i <= 34; i++) if(a[i] >= 2) tp[++cnt] = c[a[i]][2] * (b[i] ? 4 : 1);
if(cnt >= 7)
{
sort(tp + 1, tp + cnt + 1);
long long tmp = 1;
for(int i = cnt; i > cnt - 7; i--) tmp *= tp[i];
ans = max(ans, tmp * 7);
}
设\(f[i][j][k][l][0/1]\)代表已经选完了前\(i\)种牌, 有\(j\)个顺子/刻子/杠子, 以\(i\) - \(1\)开头的顺子有\(k\)个, 以\(i\)开头的顺子有\(l\)个, 有没有选雀头的最大分数.
其中, \(k\)和\(l\)都应该不大于2, 因为当\(k\)或\(l\)中有一个大于2时, 我们就可以把些看做三个杠子或者三个刻子, 这可以直接记录到\(j\)里面去.
所以我们得到下面几个转移式(这个题顺推似乎要好推一些(感谢题解, 滑稽)):
for(int i = 0; i < 34; i++)
{
for(int j = 0; j <= 4; j++)
{
for(int k = 0; k < 3 && j + k <= 4; k++)
{
if(k && (i == 9 || i == 18 || i >= 27)) break; //注意, 当i=9或i=18或i>=27时不能够由这种牌为开头组成顺子
for(int l = 0; l < 3 && j + k + l <= 4; l++)
{
if(l && (i == 9 || i == 18 || i >= 27)) break; //同上
if(!f[i][j][k][l][0] && !f[i][j][k][l][1]) continue;
for(int x = k + l; x <= a[i + 1]; x++)
{
long long tmp = c[a[i + 1]][x] * (b[i + 1] ? (1 << x) : 1);
if(x == 4 && !k && !l && j <= 3)
{
f[i + 1][j + 1][0][0][0] = max(f[i + 1][j + 1][0][0][0], f[i][j][k][l][0] * tmp);
f[i + 1][j + 1][0][0][1] = max(f[i + 1][j + 1][0][0][1], f[i][j][k][l][1] * tmp);
}
//凑一个杠子, 条件是x - k - l >= 4, 又由于x <= 4, 所以x = 4, 并且j + 1 <= 4, 这个1是凑出来的杠子, 最后的面子和杠子总和不能超过4, 由于l, k均为0, 所以第i + 1个的情况也为0, 没有以i为开头的顺子, 也没有以i + 1为开头的顺子, 全拿去凑杠子去了
if(x - k - l >= 3 && j + x - 2 <= 4)
{
f[i + 1][j + k + 1][l][x - k - l - 3][0] = max(f[i + 1][j + k + 1][l][x - k - l - 3][0], f[i][j][k][l][0] * tmp);
f[i + 1][j + k + 1][l][x - k - l - 3][1] = max(f[i + 1][j + k + 1][l][x - k - l - 3][1], f[i][j][k][l][1] * tmp);
}
//凑一个刻子, 那么应该满足的条件是x - k - l >= 3, 即在满足k与l两个要求后剩余的牌还能够拿出来凑成一个杠子, 那么最后到第i + 1种牌时以i + 1为开头的数量应该是x - k - l - 3, 凑完顺子和刻子之后剩下来的当面子, 此时杠子面子数量总和为j + k + 1
if(x - k - l >= 2 && j + x - 2 <= 4)
f[i + 1][j + k][l][x - k - l - 2][1] = max(f[i + 1][j + k][l][x - k - l - 2][1], f[i][j][k][l][0] * tmp);
//拿当前牌凑一个雀头, 前提条件是没有雀头, 类比上面的情况分析可知条件分别为x - k - l >= 2, j + x - 2 <= 4, 这里需要注意一下的是由于雀头不算在杠子和面子的总和中, 所以新增的数量便是k, 以i + 1为开头的顺子数量为x - k - l - 2
if(j + x <= 4 && x - k - l < 3)
{
f[i + 1][j + k][l][x - k - l][0] = max(f[i + 1][j + k][l][x - k - l][0], f[i][j][k][l][0] * tmp);
f[i + 1][j + k][l][x - k - l][1] = max(f[i + 1][j + k][l][x - k - l][1], f[i][j][k][l][1] * tmp);
}
//什么都不凑只拿来做顺子开头的情况, 自己类比上面分析一下即可
}
}
}
}
}
最后比较一下, 输出三种中的最大值即可, 下面是完整代码
完整代码
#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
int c[5][5], a[40], T, cnt;
int gs[14] = { 0, 1, 9, 10, 18, 19, 27, 28, 29, 30, 31, 32, 33, 34 };
long long tp[40], f[40][5][3][3][2];
bool b[40];
inline int read()
{
int x = 0, w = 1;
char c = getchar();
while(c < '0' || c > '9') { if (c == '-') w = -1; c = getchar(); }
while(c >= '0' && c <= '9') { x = x * 10 + c - '0'; c = getchar(); }
return x * w;
}
void check1(char c, char ch, int pos) { if(c == ch) a[pos]++; }
void check2(char c, char ch, int pos) { if(c == ch) b[pos] = 1; }
int main()
{
T = read();
for(int i = 0; i <= 4; i++)
for(int j = 0; j <= i; j++)
c[i][j] = (j ? c[i - 1][j] + c[i - 1][j - 1] : 1);
while(T--)
{
memset(a, 0, sizeof(a)); memset(f, 0, sizeof(f)); memset(b, 0, sizeof(b));
while(1)
{
string s; cin>>s;
if(s[0] == '0') break;
check1(s[0], 'E', 28); check1(s[0], 'S', 29); check1(s[0], 'W', 30); check1(s[0], 'N', 31); check1(s[0], 'Z', 32); check1(s[0], 'B', 33);
check1(s[0], 'F', 34); check1(s[1], 'm', s[0] - '0'); check1(s[1], 'p', s[0] - '0' + 9); check1(s[1], 's', s[0] - '0' + 18);
}
while(1)
{
string s; cin>>s;
if(s[0] == '0') break;
check2(s[0], 'E', 28); check2(s[0], 'S', 29); check2(s[0], 'W', 30); check2(s[0], 'N', 31); check2(s[0], 'Z', 32); check2(s[0], 'B', 33);
check2(s[0], 'F', 34); check2(s[1], 'm', s[0] - '0'); check2(s[1], 'p', s[0] - '0' + 9); check2(s[1], 's', s[0] - '0' + 18);
}
for(int i = 1; i <= 34; i++) a[i] = 4 - a[i];
long long ans = 0;
for(int i = 1; i <= 13; i++)
{
long long tmp = 1;
for(int j = 1; j <= 13; j++)
{
if(i == j)
{
if(a[gs[j]] < 2) tmp = 0;
else tmp *= c[a[gs[j]]][2] * (b[gs[j]] ? 4 : 1);
}
else
{
if(a[gs[j]] < 1) tmp = 0;
else tmp *= c[a[gs[j]]][1] * (b[gs[j]] ? 2 : 1);
}
}
ans = max(ans, tmp * 13);
}
cnt = 0;
for(int i = 1; i <= 34; i++) if(a[i] >= 2) tp[++cnt] = c[a[i]][2] * (b[i] ? 4 : 1);
if(cnt >= 7)
{
sort(tp + 1, tp + cnt + 1);
long long tmp = 1;
for(int i = cnt; i > cnt - 7; i--) tmp *= tp[i];
ans = max(ans, tmp * 7);
}
f[0][0][0][0][0] = 1;
for(int i = 0; i < 34; i++)
{
for(int j = 0; j <= 4; j++)
{
for(int k = 0; k < 3 && j + k <= 4; k++)
{
if(k && (i == 9 || i == 18 || i >= 27)) break;
for(int l = 0; l < 3 && j + k + l <= 4; l++)
{
if(l && (i == 9 || i == 18 || i >= 27)) break;
if(!f[i][j][k][l][0] && !f[i][j][k][l][1]) continue;
for(int x = k + l; x <= a[i + 1]; x++)
{
long long tmp = c[a[i + 1]][x] * (b[i + 1] ? (1 << x) : 1);
if(x == 4 && !k && !l && j <= 3)
{
f[i + 1][j + 1][0][0][0] = max(f[i + 1][j + 1][0][0][0], f[i][j][k][l][0] * tmp);
f[i + 1][j + 1][0][0][1] = max(f[i + 1][j + 1][0][0][1], f[i][j][k][l][1] * tmp);
}
if(x - k - l >= 3 && j + x - 2 <= 4)
{
f[i + 1][j + k + 1][l][x - k - l - 3][0] = max(f[i + 1][j + k + 1][l][x - k - l - 3][0], f[i][j][k][l][0] * tmp);
f[i + 1][j + k + 1][l][x - k - l - 3][1] = max(f[i + 1][j + k + 1][l][x - k - l - 3][1], f[i][j][k][l][1] * tmp);
}
if(x - k - l >= 2 && j + x - 2 <= 4)
f[i + 1][j + k][l][x - k - l - 2][1] = max(f[i + 1][j + k][l][x - k - l - 2][1], f[i][j][k][l][0] * tmp);
if(j + x <= 4 && x - k - l < 3)
{
f[i + 1][j + k][l][x - k - l][0] = max(f[i + 1][j + k][l][x - k - l][0], f[i][j][k][l][0] * tmp);
f[i + 1][j + k][l][x - k - l][1] = max(f[i + 1][j + k][l][x - k - l][1], f[i][j][k][l][1] * tmp);
}
}
}
}
}
}
ans = max(ans, f[34][4][0][0][1]);
printf("%lld\n", ans);
}
return 0;
}
复杂度自己算吧...
[luogu 5301][bzoj 5503] [GXOI/GZOI2019] 宝牌一大堆的更多相关文章
- 【BZOJ5503】[GXOI/GZOI2019]宝牌一大堆(动态规划)
[BZOJ5503][GXOI/GZOI2019]宝牌一大堆(动态规划) 题面 BZOJ 洛谷 题解 首先特殊牌型直接特判. 然后剩下的部分可以直接\(dp\),直接把所有可以存的全部带进去大力\(d ...
- [LOJ3084][GXOI/GZOI2019]宝牌一大堆——DP
题目链接: [GXOI/GZOI2019]宝牌一大堆 求最大值容易想到$DP$,但如果将$7$种和牌都考虑进来的话,$DP$状态不好设,我们将比较特殊的七小对和国士无双单独求,其他的进行$DP$. 观 ...
- P5301 [GXOI/GZOI2019]宝牌一大堆
题目地址:P5301 [GXOI/GZOI2019]宝牌一大堆 这里是官方题解(by lydrainbowcat) 部分分 直接搜索可以得到暴力分,因为所有和牌方案一共只有一千万左右,稍微优化一下数据 ...
- [GXOI/GZOI2019]宝牌一大堆(dp)
luogu bzoj 这个麻将题还算挺友善的,比隔壁zjoi的要好得多... 比较正常的做法是五维dp 但事实上六维dp也是完全不会被卡的 七对子选权值最高的七个,国士无双直接$13^2$暴力 ...
- luogu P5301 [GXOI/GZOI2019]宝牌一大堆
传送门 wdnm又是打麻将 首先国土无双可以直接枚举哪种牌用了\(2\)次算贡献,然后\(7\)个对子可以把每种牌的对子贡献排序,取最大的\(7\)个,剩下的牌直接暴力枚举是不行的,考虑dp,设\(f ...
- 【题解】Luogu P5301 [GXOI/GZOI2019]宝牌一大堆
原题传送门 首先先要学会麻将,然后会发现就是一个暴力dp,分三种情况考虑: 1.非七对子国士无双,设\(dp_{i,j,k,a,b}\)表示看到了第\(i\)种牌,一共有\(j\)个\(i-1\)开头 ...
- 题解 P5301 【[GXOI/GZOI2019]宝牌一大堆】
这道题除了非常恶心以外也没有什么非常让人恶心的地方 当然一定要说有的话还是有的,就是这题和咱 ZJOI 的 mahjong 真的是好像的说~ 于是就想说这道题出题人应该被 锕 掉 noteskey 整 ...
- [GXOI/GZOI2019]宝牌一大堆
感觉比ZJOI的麻将要休闲很多啊. 这个题就是一个最优化问题,没有面子的特殊牌型可以直接用复杂度较低的贪心判掉. 有面子的话就是一个经典dp.(曾经还在ZJOI写过这个毒瘤东西 大概就是存一下对子,面 ...
- [luogu 5300][bzoj 5502] [GXOI/GZOI2019] 与或和
题面 思路还是挺容易想的, 只是由于我还是太\(naive\)了一点不会做只会打暴力吧...... 题目要我们求所有子矩阵的\(and\)值之和与\(or\)值之和, 一看之下似乎不好入手, 我们慢慢 ...
随机推荐
- win32 注册表操作
创建键 RegCreateKeyEx int SetRecordVideoSavedDays(int newSavedDays) { HKEY hSubKey = NULL; LONG lRet = ...
- mysql数据库进阶篇
一.连表操作 1)为何需要连表操作 .把所有数据都存放于一张表的弊端 .表的组织结构复杂不清晰 .浪费空间 .扩展性极差 2)表设计,分析表与表之间的关系 寻找表与表之间的关系的套路 举例:emp表 ...
- ubuntu的应用中心打不开、闪退
原因没有细究,但问题已经解决简单粗暴: 1.更新列表 apt-get update apt-get dist-upgrade 2.重新安装应用中心 apt-get install --reinsta ...
- MZOJ 1345 hero
一道宽搜模版题,可写错了两个地方的我只得了56(掩面痛哭) http://10.37.2.111/problem.php?id=1345 先看看正确的 #include <bits/stdc++ ...
- 2018.12.30 poj3734 Blocks(生成函数)
传送门 生成函数入门题. 按照题意构造函数: 对于限定必须是出现偶数次的颜色:1+x22!+x44!+...=ex+e−x21+\frac {x^2}{2!}+\frac {x^4}{4!}+...= ...
- 将项目部署到 github上(部署到码云操作一样,前提是有码云账号)
来源:http://www.cnblogs.com/fengxiongZz/p/6477456.html 首先你需要自己的网页文件(俗称项目) 第一步:登录到Github上,新建一个repositor ...
- springboot aop+@interface实现日志记录
一.基本概念 1.自定义注解 自定义注解我们必须了解四个元注解,什么是元注解?元注解指作用于注解之上的元数据或者元信息,简单通俗的讲,元注解就是注解的注解 . Documented与Inherited ...
- VBA编程中的 sheet1 与 sheets(1)的区别
[自己理解]sheet1是一个专有名词,不是任何对象的属性,只能单独使用,特指代码所在工作簿的那个sheet1(和顺序无关,是固定的一个表,sheets(1)则和顺序有关). 参考资料: 1.代码中一 ...
- python3.4对已经存在的excel写入数据
#!/usr/bin/env python # -*- coding:utf-8 -*- # __author__ = "blzhu" """ pyt ...
- 代码的二次重构(开篇:JDBC连接数据库)
Java中使用JDBC连接数据库时,若是使用初级的代码,代码复用率非常低,连接过程简单来说分为以下几个步骤: 加载驱动包 准备好URL链接获取数据库连接(driver和url根据不同的数据库的不同而不 ...