[ZJOI2019]麻将(动态规划,自动机)
[ZJOI2019]麻将(动态规划,自动机)
题面
题解
先做一点小铺垫,对于一堆牌而言,我们只需要知道这\(n\)张牌分别出现的次数就行了,即我们只需要知道一个长度为\(n\)的串就可以了。
首先考虑如何判断一副牌是不是能胡。
出现了七对牌的情况很容易特判处理掉,只需要考虑第一种情况。
那么我们考虑\(dp\)来判断,设\(f[i][j][k][0/1]\)表示的当前考虑到了这个字符串的第\(i\)位,即考虑到了第\(i\)种牌,\(i-1,i,i+1\)的对子要用\(j\)次,\(i,i+1,i+2\)的对子要用\(k\)次,是否已经出现了一个对子。而这个\(dp\)值表示的是能够留下的最大的面子数量。不难发现\(j,k\)都不会超过\(2\)。
那么转移的时候相当于读进来当前的\(i\)有多少个,假设是\(x\),接下来\(x\)减去\(j+k\)组成顺子,然后枚举一下以多少\(x\)为开头组成顺子。这里再枚举一下是否用当前的\(x\)组成对子或者刻字。
我们把第一维丢掉,只考虑剩下的\(18\)个元素和最大的可能对子数,并且强制\(dp\)值不超过\(4\),最大对子数不超过\(7\)。这样子就会存在大量重复的状态,打表可得状态只有不到\(2100\)种。
那么我们可以提前把所有状态全部预处理出来,预处理对于当前的一个状态,插入后面一种牌\(x\)张的结果,这样子就构成了一个自动机,那么我们只需要从头到尾把一种状态插入进去就可以知道有没有胡牌。
那么此时我们只需要知道抽了\(i\)张之后还未胡牌的概率,全部累加就是答案。
考虑在自动机上\(dp\),设\(f[i][p][k]\)表示当前考虑到第\(i\)种牌,且当前在自动机的\(p\)位置上,前面一共抽了\(k\)张牌且还没有胡的方案数。
转移的时候枚举这张牌用了多少次,用组合数带进去进行计算,通过自动机进行状态的转移。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<map>
using namespace std;
#define MOD 998244353
#define MAX 402
void add(int &x,int y){x+=y;if(x>=MOD)x-=MOD;}
inline int read()
{
int x=0;bool t=false;char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=true,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return t?-x:x;
}
struct Data
{
int f[18],cnt;
void init(){memset(f,-1,sizeof(f));f[0]=cnt=0;}
bool check()
{
if(cnt>=7)return true;
for(int i=0;i<3;++i)
for(int j=0;j<3;++j)
if(f[9+i*3+j]>=4)return true;
return false;
}
}QwQ,ST[2100];
bool operator<(Data a,Data b)
{
if(a.cnt!=b.cnt)return a.cnt<b.cnt;
for(int i=0;i<18;++i)if(a.f[i]!=b.f[i])return a.f[i]<b.f[i];
return false;
}
Data Trans(Data a,int b)
{
Data c;c.init();c.cnt=min(a.cnt+(b>=2),7);
for(int i=0;i<3;++i)
for(int j=0;j<3;++j)
{
if(~a.f[i*3+j])
{
for(int k=0;k<3&&i+j+k<=b;++k)
c.f[j*3+k]=max(c.f[j*3+k],min(a.f[i*3+j]+i+(b-i-j-k>=3),4));
if(b>=2)
for(int k=0;k<3&&i+j+k<=b-2;++k)
c.f[9+j*3+k]=max(c.f[9+j*3+k],min(a.f[i*3+j]+i,4));
}
if(~a.f[9+i*3+j])
{
for(int k=0;k<3&&i+j+k<=b;++k)
c.f[9+j*3+k]=max(c.f[9+j*3+k],min(a.f[9+i*3+j]+i+(b-i-j-k>=3),4));
}
}
return c;
}
map<Data,int> M;int tot;
void Build(Data x)
{
if(x.check())return;
if(M.find(x)!=M.end())return;
ST[M[x]=++tot]=x;
for(int i=0;i<=4;++i)Build(Trans(x,i));
}
int jc[MAX],jv[MAX],inv[MAX];
int f[2][2100][MAX];
int C(int n,int m){return 1ll*jc[n]*jv[m]%MOD*jv[n-m]%MOD;}
int n,ans,s[MAX],tr[2100][5];
int main()
{
QwQ.init();Build(QwQ);
jc[0]=jv[0]=inv[0]=inv[1]=1;
for(int i=2;i<MAX;++i)inv[i]=1ll*inv[MOD%i]*(MOD-MOD/i)%MOD;
for(int i=1;i<MAX;++i)jc[i]=1ll*jc[i-1]*i%MOD;
for(int i=1;i<MAX;++i)jv[i]=1ll*jv[i-1]*inv[i]%MOD;
n=read();for(int i=1;i<=13;++i)s[read()]+=1,read();
for(int i=1;i<=tot;++i)for(int j=0;j<=4;++j)tr[i][j]=M[Trans(ST[i],j)];
f[0][1][0]=1;
for(int i=1,ss=0,nw=1,pw=0;i<=n;ss+=s[i],++i,nw^=1,pw^=1)
{
memset(f[nw],0,sizeof(f[nw]));
for(int j=1;j<=tot;++j)
for(int k=s[i];k<=4;++k)
{
if(!tr[j][k])continue;
int w=1ll*C(4-s[i],k-s[i])*jc[k-s[i]]%MOD;
for(int l=0;l<=n*4-k;++l)
if(f[pw][j][l])
add(f[nw][tr[j][k]][k+l],1ll*f[pw][j][l]*w%MOD*C(k+l-ss-s[i],k-s[i])%MOD);
}
}
for(int i=13,val=1;i<=n*4;val=1ll*val*inv[n*4-i]%MOD,++i)
{
int ret=0;
for(int j=1;j<=tot;++j)add(ret,f[n&1][j][i]);
add(ans,1ll*ret*val%MOD);
}
printf("%d\n",ans);
return 0;
}
[ZJOI2019]麻将(动态规划,自动机)的更多相关文章
- [ZJOI2019]麻将
这是一道麻将自动机的模板题(雾 其实这是一道dp套dp借助自动机实现的麻将好题! 首先把期望转化一下,拆成sigema p(x>i) 现在要计算i张牌不胡的概率,也就等价于计算i张牌不胡的方案数 ...
- 【洛谷5279】[ZJOI2019] 麻将(“胡牌自动机”上DP)
点此看题面 大致题意: 给你13张麻将牌,问你期望再摸多少张牌可以满足存在一个胡的子集. 似乎ZJOI2019Day1的最大收获是知道了什么是胡牌? 一个显然的性质 首先我们要知道一个显然的性质,即对 ...
- [ZJOI2019]麻将(DP+有限状态自动机)
首先只需要考虑每种牌出现的张数即可,然后判断一副牌是否能胡,可以DP一下,令f[i][j][k][0/1]表示到了第i位,用j次i-1,i,i+1和k次i,i+1,i+2,是否出现对子然后最大的面子数 ...
- Luogu P5279 [ZJOI2019]麻将
ZJOI2019神题,间接送我退役的神题233 考场上由于T2写挂去写爆搜的时候已经没多少时间了,所以就写挂了233 这里不多废话直接开始讲正解吧,我们把算法分成两部分 1.建一个"胡牌自动 ...
- 【题解】Luogu P5279 [ZJOI2019]麻将
原题传送门 希望这题不会让你对麻将的热爱消失殆尽 我们珂以统计每种牌出现的次数,不需要统计是第几张牌 判一副牌能不能和,类似这道题 对于这题: 设\(f[i][j][k][0/1]\)表示前\(i\) ...
- 洛谷P5279 [ZJOI2019]麻将
https://www.luogu.org/problemnew/show/P5279 以下为个人笔记,建议别看: 首先考虑如何判一个牌型是否含有胡的子集.先将牌型表示为一个数组num,其中num[i ...
- 洛谷 P5279 - [ZJOI2019]麻将(dp 套 dp)
洛谷题面传送门 一道 dp 套 dp 的 immortal tea 首先考虑如何判断一套牌是否已经胡牌了,考虑 \(dp\).我们考虑将所有牌按权值大小从大到小排成一列,那我们设 \(dp_ ...
- 洛谷P5279 [ZJOI2019]麻将(乱搞+概率期望)
题面 传送门 题解 看着题解里一堆巨巨熟练地用着专业用语本萌新表示啥都看不懂啊--顺便\(orz\)余奶奶 我们先考虑给你一堆牌,如何判断能否胡牌 我们按花色大小排序,设\(dp_{0/1,i,j,k ...
- 题解 洛谷 P5279 【[ZJOI2019]麻将】
这题非常的神啊...蒟蒻来写一篇题解. Solution 首先考虑如何判定一副牌是否是 "胡" 的. 不要想着统计个几个值 \(O(1)\) 算,可以考虑复杂度大一点的. 首先先把 ...
随机推荐
- jupyter notebook安装、登录
pip install jupyter 提示pip需要升级(本人装的是anaconda) 输入:python -m pip install --upgrade pip 安装完成. 运行jupyter ...
- 轨迹系列1——一种基于路网图层的GPS轨迹优化方案
文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/ 1.背景 GPS数据正常情况下有20M左右的偏移,在遇到高楼和桥梁等情况 ...
- Github:failed to add file / to index
我把Test项目上传到github上,为了截一部分图,来写博客.所以我就上传成功之后,把仓库Respository Test删除了,但是当我再次上传的时候,发现上传不上,会提示failed to ad ...
- Android系统的三种分屏显示模式
Google在Android 7.0中引入了一个新特性——多窗口支持,允许用户一次在屏幕上打开两个应用.在手持设备上,两个应用可以在"分屏"模式中左右并排或上下并排显示.在电视设备 ...
- Kasaraju算法--强连通图遍历及其python实现
在理解有向图和强连通分量前必须理解与其对应的两个概念,连通图(无向图)和连通分量. 连通图的定义是:如果一个图中的任何一个节点可以到达其他节点,那么它就是连通的. 例如以下图形: 这是最简单的一个连通 ...
- DAS、SAN和NAS三种存储方式
DAS存储 DAS存储在我们生活中是非常常见的,尤其是在中小企业应用中,DAS是最主要的应用模式,存储系统被直连到应用的服务器中,在中小企业中,许多的数据应用是必须安装在直连的DAS存储器上. DAS ...
- Mockito单元测试
Mockito简介 Mockito是一个单元测试框架,需要Junit的支持.在我们的项目中,都存在相当多的依赖关系,当我们在测试某一个业务相关的接口或则方法时,绝大多数时候是没有办法或则很难去添加所有 ...
- DRF 序列化器-Serializer (2)
作用 1. 序列化,序列化器会把模型对象转换成字典,经过response以后变成json字符串 2. 完成数据校验功能 3. 反序列化,把客户端发送过来的数据,经过request以后变成字典,序列化器 ...
- 【Python 补充01】Python运算符
Python运算符 举个简单的例子 4 +5 = 9 . 例子中,4 和 5 被称为操作数,"+" 称为运算符. 1.算术运算符 + - * / # 加减乘除 % # 取模(返回除 ...
- 正益工作能担起PaaS+SaaS的未来探索吗?
没有竞争,行业没有未来.不参与竞争,企业没有未来.中国企业的类型纷繁复杂,也决定了企业的多样化需求.云计算和移动化的双重叠加,企业管理需要重新梳理,企业业务创新日益频繁,个性化需求日益突出,软件服务商 ...