这是一道麻将自动机的模板题(雾

其实这是一道dp套dp借助自动机实现的麻将好题!

首先把期望转化一下,拆成sigema p(x>i)

现在要计算i张牌不胡的概率,也就等价于计算i张牌不胡的方案数。

如果我们能建立一个关于麻将的自动机,支持插入麻将,判断当前牌型是否能胡,既可以在麻将自动机上dp解决本题。

先考虑如何压缩一副麻将,显然只需要把它的每一种牌型的个数算出来即可。

再考虑如何判定一副麻将是否能胡,这个可以借助dp。

dp[i][j][k]表示当前有i个x,还额外预留了j个(x,x-1),k表示是否有对子。

转移的时候枚举下一个位置预留几组,走个顺子即可。

现在我们要对这个过程建立自动机。

考虑两个麻将状态不同,当且仅当它们存在某种相同的状态,dp值却互不相同。

写一个结构体,然后bfs寻找状态+map判重即可。

不同的状态数不多,2000多种的样子。

建好这个东西后再去dp就可以了。

#include<bits/stdc++.h>
#define N 1100
#define M 5000
#define db double
#define ll long long
#define ldb long double
using namespace std;
inline int read()
{
char ch=0;
int x=0,flag=1;
while(!isdigit(ch)){ch=getchar();if(ch=='-')flag=-1;}
while(isdigit(ch)){x=(x<<3)+(x<<1)+ch-'0';ch=getchar();}
return x*flag;
}
const int mo=998244353,inf=1e9+7;
struct node
{
int cnt,f[3][3][2];
void clear()
{
cnt=0;
for(int i=0;i<3;i++)for(int j=0;j<3;j++)for(int k=0;k<2;k++)f[i][j][k]=-inf;
}
};
bool operator<(node a,node b)
{
for(int i=0;i<3;i++)for(int j=0;j<3;j++)for(int k=0;k<2;k++)
if(a.f[i][j][k]!=b.f[i][j][k])return a.f[i][j][k]<b.f[i][j][k];
return a.cnt<b.cnt;
}
bool operator==(node a,node b)
{
for(int i=0;i<3;i++)for(int j=0;j<3;j++)for(int k=0;k<2;k++)
if(a.f[i][j][k]!=b.f[i][j][k])return false;
if(a.cnt!=b.cnt)return false;
return true;
}
node goal;
void up(int &x,int k){x=max(x,k);}
node operator+(node v,int t)
{
if(v==goal)return goal;
static node ans;ans.clear();
ans.cnt=v.cnt+(t>=2);
if(ans.cnt>=7)return goal;
for(int i=0;i<3;i++)for(int j=0;j<3;j++)for(int k=0;k<2;k++)
{
int o=v.f[i][j][k];
if(o<0)continue;
for(int a=0;a<=min(i,t);a++)//枚举预留组数
for(int b=0;b<=min(j,t);b++)//枚举走几个顺子
if(a+b<=t)
{
for(int c=0;c<=1;c++)//枚举走几个对子
for(int d=0;d<=1;d++)//枚举走几个面子
if(a+b+c*2+d*3<=t)up(ans.f[min(2,t-a-b-c*2-d*3)][a][k|c],o+b+d);
}
}
for(int i=0;i<3;i++)for(int j=0;j<3;j++)for(int k=0;k<2;k++)
{
ans.f[i][j][k]=min(ans.f[i][j][k],4);
if(k==1&&ans.f[i][j][k]==4)return goal;
}
return ans;
}
node qwq[M];
queue<node>q;
map<node,int>mp;
int size,cnt[N],fac[N],C[N][N],dp[N][M],DP[N][M],nxt[M][10];
void build()
{
node st;
st.clear();goal.clear();
st.f[0][0][0]=0;goal.cnt=-1;
for(int i=0;i<3;i++)for(int j=0;j<3;j++)for(int k=0;k<2;k++)goal.f[i][j][k]=-mo;
mp[st]=++size;mp[goal]=++size;q.push(st);
for(int i=0;i<=4;i++)nxt[2][i]=2;
while(!q.empty())
{
node x=q.front();q.pop();
qwq[mp[x]]=x;
for(int i=0;i<=4;i++)
{
node to=x+i;
if(!mp.count(to))mp[to]=++size,q.push(to);
nxt[mp[x]][i]=mp[to];
}
}
}
int ksm(int x,int k)
{
int ans=1;
while(k){if(k&1)ans=1ll*ans*x%mo;k>>=1;x=1ll*x*x%mo;}
return ans;
}
int main()
{
build();
int n=read(),ans=0;
for(int i=1;i<=13;i++){cnt[read()]++;read();}
fac[0]=C[0][0]=1;
for(int i=1;i<=4*n;i++)
{
C[i][0]=1;fac[i]=1ll*fac[i-1]*i%mo;
for(int j=1;j<=i;j++)C[i][j]=(C[i-1][j-1]+C[i-1][j])%mo;
}
DP[0][1]=1;
for(int o=1;o<=n;o++)
{
for(int i=0;i<=4*o;i++)
for(int x=1;x<=size;x++)
dp[i][x]=DP[i][x],DP[i][x]=0;
for(int i=0;i<=4*o;i++)
for(int x=1;x<=size;x++)
{
if(!dp[i][x])continue;
for(int k=cnt[o];k<=4;k++)
{
int to=nxt[x][k];if(to==2)continue;
DP[i+k][to]=(DP[i+k][to]+(1ll*C[4-cnt[o]][k-cnt[o]]*dp[i][x]%mo))%mo;
}
}
}
for(int i=0;i<4*n;i++)
{
int tot=0;
for(int x=1;x<=size;x++)tot=(tot+DP[i][x])%mo;
tot=1ll*tot*fac[i-13]%mo*fac[4*n-i]%mo;
ans=(ans+1ll*tot*ksm(fac[4*n-13],mo-2))%mo;
}
printf("%d",(ans%mo+mo)%mo);
return 0;
}

[ZJOI2019]麻将的更多相关文章

  1. [ZJOI2019]麻将(动态规划,自动机)

    [ZJOI2019]麻将(动态规划,自动机) 题面 洛谷 题解 先做一点小铺垫,对于一堆牌而言,我们只需要知道这\(n\)张牌分别出现的次数就行了,即我们只需要知道一个长度为\(n\)的串就可以了. ...

  2. Luogu P5279 [ZJOI2019]麻将

    ZJOI2019神题,间接送我退役的神题233 考场上由于T2写挂去写爆搜的时候已经没多少时间了,所以就写挂了233 这里不多废话直接开始讲正解吧,我们把算法分成两部分 1.建一个"胡牌自动 ...

  3. 【题解】Luogu P5279 [ZJOI2019]麻将

    原题传送门 希望这题不会让你对麻将的热爱消失殆尽 我们珂以统计每种牌出现的次数,不需要统计是第几张牌 判一副牌能不能和,类似这道题 对于这题: 设\(f[i][j][k][0/1]\)表示前\(i\) ...

  4. 洛谷P5279 [ZJOI2019]麻将(乱搞+概率期望)

    题面 传送门 题解 看着题解里一堆巨巨熟练地用着专业用语本萌新表示啥都看不懂啊--顺便\(orz\)余奶奶 我们先考虑给你一堆牌,如何判断能否胡牌 我们按花色大小排序,设\(dp_{0/1,i,j,k ...

  5. 【洛谷5279】[ZJOI2019] 麻将(“胡牌自动机”上DP)

    点此看题面 大致题意: 给你13张麻将牌,问你期望再摸多少张牌可以满足存在一个胡的子集. 似乎ZJOI2019Day1的最大收获是知道了什么是胡牌? 一个显然的性质 首先我们要知道一个显然的性质,即对 ...

  6. 洛谷P5279 [ZJOI2019]麻将

    https://www.luogu.org/problemnew/show/P5279 以下为个人笔记,建议别看: 首先考虑如何判一个牌型是否含有胡的子集.先将牌型表示为一个数组num,其中num[i ...

  7. [ZJOI2019]麻将(DP+有限状态自动机)

    首先只需要考虑每种牌出现的张数即可,然后判断一副牌是否能胡,可以DP一下,令f[i][j][k][0/1]表示到了第i位,用j次i-1,i,i+1和k次i,i+1,i+2,是否出现对子然后最大的面子数 ...

  8. 题解 洛谷 P5279 【[ZJOI2019]麻将】

    这题非常的神啊...蒟蒻来写一篇题解. Solution 首先考虑如何判定一副牌是否是 "胡" 的. 不要想着统计个几个值 \(O(1)\) 算,可以考虑复杂度大一点的. 首先先把 ...

  9. 洛谷 P5279 - [ZJOI2019]麻将(dp 套 dp)

    洛谷题面传送门 一道 dp 套 dp 的 immortal tea 首先考虑如何判断一套牌是否已经胡牌了,考虑 \(dp\)​​​​​.我们考虑将所有牌按权值大小从大到小排成一列,那我们设 \(dp_ ...

随机推荐

  1. 新增职责 不能从IE进入的问题 此责任无可用函数 (转)

    此责任无可用函数(The Function Is Not Available Under The Responsibility) When attempting to navigate to a fu ...

  2. openwrt路由器进入安全模式

    openwrt路由器型号:WNDR3800 一.实验背景 在pc机上通过xshell软件登录openwrt路由器,pc机通过网线与openwrt路由器的LAN接口相连.openwrt路由器自带两块无线 ...

  3. # 2017-2018-2 20155228 《信息安全系统设计原理》 使用VirtualStudio2008创建和调用静态库和使用VirtualC++6.0创建和调用动态库

    使用virtual c++ 6.0创建和调用动态库 不得不说一下关于环境的问题 只要我打一个响指,一半的安装在win7上的VC6.0都会因为兼容性问题直接崩掉 懒得研究怎么解决兼容性的问题了,直接开一 ...

  4. Class打包成jar

    Class打包成jar 现在我的文件夹的目录在: C:\Users\linsenq\Desktop\cglibjar 我要把位于这个目录下的所有文件夹以及这个文件夹下的.class文件打成jar包 第 ...

  5. Tomcat 控制台UTF-8乱码问题

    1.修改cmd的编码格式 快捷键win+R打开运行程序,输入regedit打开注册表,找到以下路劲并且修改. [HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Comman ...

  6. js中的数组方法

    数组的方法有数组原型方法,也有从object对象继承来的方法,这里我们只介绍数组的原型方法,数组原型方法主要有以下这些: join()push()和pop()shift() 和 unshift()so ...

  7. LeetCode Weekly Contest 121

    上周因为感冒没有刷题,两个星期没有刷题,没手感了,思维也没有那么活跃了,只刷了一道,下个星期努力. 984. String Without AAA or BBB Given two integers  ...

  8. windows下手动安装composer并配置环境变量

    windows下手动安装composer并配置环境变量   转载地址: https://my.oschina.net/7sites/blog/209997 之前发表过一篇如何为composer设置代理 ...

  9. 自动弹出pickerview

    UIPickerView是开发中常用的控件,日期选择.年龄选择.城市的多级联动等等都会使用,它一般是在点击某个按钮后出现,展现方式和UITextView一样,从页面底部弹出,选中后或者点击控件以外区域 ...

  10. 单例模式-懒汉式的一次多线程Debug

    单例模式要要点就是一个类只会存在一个实例,要想达到这种效果,最重要的就是将构造方法设置为私有,然后通过static的方法来获取对象. 上述设计并不线程安全,因为在lazySingleton = new ...