【BZOJ5503】[GXOI/GZOI2019]宝牌一大堆(动态规划)

题面

BZOJ

洛谷

题解

首先特殊牌型直接特判。

然后剩下的部分可以直接\(dp\),直接把所有可以存的全部带进去大力\(dp\)就行了。

发现每多一张牌胡的本质就是把一个刻字换成杠子,所以这两个东西记录在一起就行了。

那么状态就是\(f[i][0/1/2/3/4][0/1/2][0/1/2][0/1]\)

分别表示刻字、杠子、顺子的数量,\(i-1,i,i+1\)的顺子数量,\(i,i+1,i+2\)的顺子的数量,以及是否已经有对子。

转移比较容易。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<vector>
using namespace std;
#define ll long long
#define MAX 140
int Read()
{
char ch[3];scanf("%s",ch);
if(ch[0]=='E')return 28;if(ch[0]=='S')return 29;
if(ch[0]=='W')return 30;if(ch[0]=='N')return 31;
if(ch[0]=='Z')return 32;if(ch[0]=='B')return 33;
if(ch[0]=='F')return 34;if(ch[0]=='0')return 0;
int x=ch[0]-48;if(ch[1]=='p')x+=9;if(ch[1]=='s')x+=18;
return x;
}
ll C[MAX][MAX];
int n,s[MAX];bool book[MAX];
ll f[35][6][3][3][2];
int pre[14]={0,1,9,10,18,19,27,28,29,30,31,32,33,34};
int bin[10]={1,2,4,8,16,32,64,128,256,512};
void cmax(ll &x,ll y){x=max(x,y);}
int main()
{
for(int i=0;i<10;++i)C[i][0]=1;
for(int i=1;i<10;++i)
for(int j=1;j<=i;++j)C[i][j]=C[i-1][j]+C[i-1][j-1];
int T,x;scanf("%d",&T);
while(T--)
{
for(int i=1;i<=34;++i)s[i]=4,book[i]=false;
while((x=Read()))s[x]-=1;
while((x=Read()))book[x]=true;
ll ans=0;
//Task1
{
bool fl=true;int mx=0;ll ret=1;
for(int i=1;i<=13;++i)
if(!s[pre[i]]){fl=false;break;}
else mx=max(mx,(s[pre[i]]-1)*(book[pre[i]]?2:1)),ret=ret*(book[pre[i]]?2:1)*s[pre[i]];
if(fl)ret=ret*mx*13/2,ans=max(ans,ret);
}
//Task2
{
vector<int> val;
for(int i=1;i<=34;++i)
if(s[i]>=2)val.push_back((book[i]?4:1)*(s[i]*(s[i]-1)/2));
sort(val.begin(),val.end());
if(val.size()>=7)
{
ll ret=1;
for(int i=7;i;--i)ret*=val.back(),val.pop_back();
ans=max(ans,ret*7);
}
}
//Task3
{
memset(f,0,sizeof(f));
f[0][0][0][0][0]=1;
for(int i=1;i<=34;++i)
for(int j=0;j<5;++j)
for(int k=0;k<3&&j+k<=4&&k<=s[i];++k)
{
if(k>0&&(i==10||i==19||i>=28||i==11||i==20))break;
for(int l=0;l<3&&j+k+l<=4&&k+l<=s[i];++l)
{
if(l>0&&(l==9||i==18||i==27||i==10||i==19||i>=28))break;
if(!f[i-1][j][k][l][0]&&!f[i-1][j][k][l][1])continue;
for(int m=0;k+l+m<=s[i]&&m<3;++m)
{
if(m>0&&(i==9||i==18||i>=27))continue;
if(s[i]>=4+k+l+m)
{
cmax(f[i][j+1+k][l][m][0],(book[i]?16:1)*f[i-1][j][k][l][0]);
cmax(f[i][j+1+k][l][m][1],(book[i]?16:1)*f[i-1][j][k][l][1]);
}
if(s[i]>=3+k+l+m)
{
cmax(f[i][j+1+k][l][m][0],(book[i]?bin[3+k+l+m]:1)*C[s[i]][3+k+l+m]*f[i-1][j][k][l][0]);
cmax(f[i][j+1+k][l][m][1],(book[i]?bin[3+k+l+m]:1)*C[s[i]][3+k+l+m]*f[i-1][j][k][l][1]);
}
if(s[i]>=2+k+l+m)
{
cmax(f[i][j+k][l][m][1],(book[i]?bin[2+k+l+m]:1)*C[s[i]][2+k+l+m]*f[i-1][j][k][l][0]);
}
if(s[i]>=k+l+m)
{
cmax(f[i][j+k][l][m][0],(book[i]?bin[k+l+m]:1)*C[s[i]][k+l+m]*f[i-1][j][k][l][0]);
cmax(f[i][j+k][l][m][1],(book[i]?bin[k+l+m]:1)*C[s[i]][k+l+m]*f[i-1][j][k][l][1]);
}
}
}
}
ans=max(ans,f[34][4][0][0][1]);
}
printf("%lld\n",ans);
}
}

【BZOJ5503】[GXOI/GZOI2019]宝牌一大堆(动态规划)的更多相关文章

  1. [LOJ3084][GXOI/GZOI2019]宝牌一大堆——DP

    题目链接: [GXOI/GZOI2019]宝牌一大堆 求最大值容易想到$DP$,但如果将$7$种和牌都考虑进来的话,$DP$状态不好设,我们将比较特殊的七小对和国士无双单独求,其他的进行$DP$. 观 ...

  2. P5301 [GXOI/GZOI2019]宝牌一大堆

    题目地址:P5301 [GXOI/GZOI2019]宝牌一大堆 这里是官方题解(by lydrainbowcat) 部分分 直接搜索可以得到暴力分,因为所有和牌方案一共只有一千万左右,稍微优化一下数据 ...

  3. [GXOI/GZOI2019]宝牌一大堆(dp)

    luogu     bzoj 这个麻将题还算挺友善的,比隔壁zjoi的要好得多... 比较正常的做法是五维dp 但事实上六维dp也是完全不会被卡的 七对子选权值最高的七个,国士无双直接$13^2$暴力 ...

  4. 题解 P5301 【[GXOI/GZOI2019]宝牌一大堆】

    这道题除了非常恶心以外也没有什么非常让人恶心的地方 当然一定要说有的话还是有的,就是这题和咱 ZJOI 的 mahjong 真的是好像的说~ 于是就想说这道题出题人应该被 锕 掉 noteskey 整 ...

  5. luogu P5301 [GXOI/GZOI2019]宝牌一大堆

    传送门 wdnm又是打麻将 首先国土无双可以直接枚举哪种牌用了\(2\)次算贡献,然后\(7\)个对子可以把每种牌的对子贡献排序,取最大的\(7\)个,剩下的牌直接暴力枚举是不行的,考虑dp,设\(f ...

  6. [GXOI/GZOI2019]宝牌一大堆

    感觉比ZJOI的麻将要休闲很多啊. 这个题就是一个最优化问题,没有面子的特殊牌型可以直接用复杂度较低的贪心判掉. 有面子的话就是一个经典dp.(曾经还在ZJOI写过这个毒瘤东西 大概就是存一下对子,面 ...

  7. [luogu 5301][bzoj 5503] [GXOI/GZOI2019] 宝牌一大堆

    题面 好像ZJOI也考了一道麻将, 这是要发扬中华民族的赌博传统吗??? 暴搜都不会打, 看到题目就自闭了, 考完出来之后看题解, \(dp\), 可惜自己想不出来... 对于国士无双(脑子中闪过了韩 ...

  8. 【题解】Luogu P5301 [GXOI/GZOI2019]宝牌一大堆

    原题传送门 首先先要学会麻将,然后会发现就是一个暴力dp,分三种情况考虑: 1.非七对子国士无双,设\(dp_{i,j,k,a,b}\)表示看到了第\(i\)种牌,一共有\(j\)个\(i-1\)开头 ...

  9. [GX/GZOI2019]宝牌一大堆(DP)

    出这种麻将题有意思吗? 乍看很难实则很水,就是麻将式DP,想必大家很熟悉了吧.首先把“国士无双”和“七对子”两种牌型判掉,然后观察牌胡的形式,发现每多一张牌实际上就是把1个面子变成1个杠子,然后可以直 ...

随机推荐

  1. 加载loading对话框的功能(不退出沉浸式效果)

    上一篇基于修改系统源码的前提下,实现了完全的沉浸式体验效果.可参考这篇 戳这 一.自定义Dialog 在沉浸式效果下,当界面弹出对话框时,对话框将获取到焦点,这将导致界面退出沉浸式效果,那么是不是能通 ...

  2. Vue之虚拟DOM

    一.真实DOM和其解析流程? 浏览器渲染引擎工作流程都差不多,大致分为5步,创建DOM树——创建StyleRules——创建Render树——布局Layout——绘制Painting 第一步,用HTM ...

  3. Python第五天 文件访问 for循环访问文件 while循环访问文件 字符串的startswith函数和split函数 linecache模块

    Python第五天   文件访问    for循环访问文件    while循环访问文件   字符串的startswith函数和split函数  linecache模块 目录 Pycharm使用技巧( ...

  4. Redis(十一):Redis的事务功能详解

    相关命令 1. MULTI 用于标记事务块的开始.Redis会将后续的命令逐个放入队列中,然后才能使用EXEC命令原子化地执行这个命令序列. 这个命令的运行格式如下所示: MULTI 这个命令的返回值 ...

  5. C#的自动拼接Sql语句Insert方法及思路

    思路: 1.想想插入语句,大概是这样的一个框架:INSERT INTO 表名 (数据库列名) values (值) 2.这里要3个变量是不固定的,分别是:表名.数据库列名.值: a.表名我们这里很容易 ...

  6. 宋宝华:关于Ftrace的一个完整案例【转】

    Ftrace简介 Ftrace是Linux进行代码级实践分析最有效的工具之一,比如我们进行一个系统调用,出来的时间过长,我们想知道时间花哪里去了,利用Ftrace就可以追踪到一级级的时间分布. Ftr ...

  7. jQuery中toggle与slideToggle以及fadeToggle之间的不同

    toggle()方法: 定义和用法 切换元素的可见状态.如果被选元素可见,则隐藏这些元素,如果被选元素隐藏,则显示这些元素. 语法: $(selector).toggle(speed,callback ...

  8. Win10 Service'MongoDB Server' failed to start. Verify that you have sufficient privileges to start system services【简记】

    最近工作中有需要用到 MongoDB数据库,以前用的3.*的版本,这次用的是较新4.0.6的版本,然后去官网下载安装. 安装到一半,就弹出如下提示,说是"MongoDB Server&quo ...

  9. java格式化

    http://tool.oschina.net/apidocs/apidoc?api=jdk-zh https://www.jianshu.com/p/c8f16cab35e1# 参考官方的 api说 ...

  10. oc中的委托模式

    通过一个例子来理解委托模式 首先定义个协议 协议(protocol) :它可以声明一些必须实现的方法和选择实现的方法  (在java中称为接口) // // StudentDelegate.h // ...