LGP5279题解
这题好牛逼啊。。。
虽然说也是 DP 套 DP,但是感觉比 TJOI 那题高明到哪里去了(
我们先考虑如何计算期望。如果设 \(f_i\) 为拿到 \(i\) 张牌后胡的方案数,这个并不是很好做,因为你要考虑前面不胡。
所以我们设 \(f_i\) 为手上有 \(13+i\) 张牌但还没有胡的方案数,那么答案就是:
\]
发现一个手上牌的顺序不影响是否胡,所以有两个阶乘用来打乱排序(因为牌与牌直接互不相同)。
我们考虑外层的 DP 建立一个“胡牌自动机”,所以需要写一个用来判定是否胡牌的 DP。
加入新的牌就是在自动机上面走一步。我们设 \(dp_{i,x,y,0/1}\) 为加入了前 \(i\) 类牌,现在手上有 \(x\) 对 \((i-1,i)\) 的牌和 \(y\) 张 \(i\),\(0/1\) 表示当前无/有对子。表示的是在这个状态下面子最多个数。
我们将发现如果 \(x,y\geq 3\),那么可以自动对出来一个面子,所以有 \(0\leq x,y<3\)。
我们把这两个维度写到矩阵上面。然后交换维度变成 \(dp_{i,0/1,x,y}\)。
考虑加入 \(x\) 张牌时如何从 \(dp_{i-1}\) 转移到 \(dp_i\)。
考虑加入 \(x\) 张牌的时候,先凑到 \((i-2,i-1)\) 里面去,然后自己和自己凑面子,最后加上已经有的面子个数。
如果 \(x\geq 2\),那么我们可以砍掉一对对子再加进去。
也就是:\((f_{i-1,0},x)\to f_{i,0},(f_{i-1,1},x)\to f_{i,1},(f_{i-1,0},x-2)\to f_{i,1}\)。
然后我们再额外记录对子个数即可。
然后我们用一个 std::map 离散化一下就可以建出“胡牌自动机”了。
这个是主要的,在这个 DFA 上面 DP 反而很简单。
直接考虑当前加 \(x\) 张牌,然后从牌堆里面把这 \(x\) 张牌选出来,丢到下一个 DFA 节点里面去就可以了。
我们可以把所有胡掉的节点压成一个节点,然后标记 \(t=-1\),可以节省空间。
#include<cstdio>
#include<map>
typedef int ui;
const ui M=105,mod=998244353;
const ui C[5][5]={{1,0,0,0,0},{1,1,0,0,0},{1,2,1,0,0},{1,3,3,1,0},{1,4,6,4,1}};
ui n,m,a[M],fac[M*4],ifac[M*4],dp[2][M*4][2100];
inline ui min(const ui&a,const ui&b){
return a>b?b:a;
}
inline void max(ui&a,const ui&b){
if(b>a)a=b;
}
struct HuAutomation{
ui tot;
struct Matrix{
#define For for(ui i=0;i^3;++i)for(ui j=0;j^3;++j)
ui p[3][3];
Matrix(){
For p[i][j]=-1;
}
inline ui*operator[](const ui&x){
return p[x];
}
inline bool operator==(Matrix q)const{
For if(p[i][j]!=q[i][j])return false;return true;
}
inline bool operator<(Matrix q)const{
For if(p[i][j]!=q[i][j])return p[i][j]<q[i][j];return false;
}
inline bool CheckHu()const{
For if(~p[i][j]&&p[i][j]>3)return true;return false;
}
inline void Update(Matrix bel,const ui&x){
For if(~bel[i][j])for(ui k=0;k^3&&i+j+k<=x;++k){
max(p[j][k],min(i+bel[i][j]+(x-i-j-k)/3,4));
}
}
#undef For
};
struct Node{
ui t;Matrix P[2];
inline Node(){
t=0;P[0]=P[1]=Matrix();
}
inline bool operator<(Node it)const{
return t==it.t?P[0]==it.P[0]?P[1]==it.P[1]?0:P[1]<it.P[1]:P[0]<it.P[0]:t<it.t;
}
inline Node Hu()const{
Node x;x.t=-1;return x;
}
inline bool CheckHu()const{
return!~t||t>=7||P[1].CheckHu();
}
inline Node operator+(const ui&x){
if(CheckHu())return Hu();Node ans;
ans.P[0].Update(P[0],x);ans.P[1].Update(P[1],x);ans.t=t;
if(x>=2)ans.P[1].Update(P[0],x-2),++ans.t;
if(ans.CheckHu())ans=Hu();
return ans;
}
}tr[2100];
ui trans[2100][5];
inline ui*operator[](const ui&x){
return trans[x];
}
inline Node Hu()const{
return tr[0].Hu();
}
std::map<Node,ui>P;
#define pos(x) (P.count(x)?P[x]:(tr[P[x]=++tot]=x,tot))
inline void Build(){
tr[1].P[0][0][0]=0;P[tr[1]]=++tot;P[tr[2]=Hu()]=++tot;
for(ui i=1;i<=tot;++i)if(i!=2){
for(ui t=0;t<=4;++t)trans[i][t]=pos(tr[i]+t);
}
}
}Hu;
signed main(){
ui ans(0);
Hu.Build();scanf("%u",&n);m=(n<<2)-13;dp[0][0][1]=1;
fac[0]=fac[1]=ifac[0]=ifac[1]=1;
for(ui i=2;i<=m;++i)fac[i]=1ull*fac[i-1]*i%mod,ifac[i]=1ull*(mod-mod/i)*ifac[mod%i]%mod;
for(ui i=2;i<=m;++i)ifac[i]=1ull*ifac[i-1]*ifac[i]%mod;
for(ui i=1;i<=13;++i){
ui w,t;scanf("%u%u",&w,&t);++a[w];
}
for(ui i=0;i<n;++i){
const ui&q=i&1,p=q^1;
for(ui j=0;j<=m;++j)for(ui S=1;S<=Hu.tot;++S)if(dp[q][j][S]){
for(ui x=0;x<=4-a[i+1];++x){
dp[p][j+x][Hu[S][a[i+1]+x]]=(dp[p][j+x][Hu[S][a[i+1]+x]]+1ull*C[4-a[i+1]][x]*dp[q][j][S])%mod;
}
}
for(ui i=0;i<=m;++i)for(ui S=1;S<=Hu.tot;++S)dp[q][i][S]=0;
}
for(ui i=1;i<=m;++i)for(ui S=1;S<=Hu.tot;++S)if(S!=2){
ans=(ans+1ull*fac[i]*fac[m-i]%mod*dp[n&1][i][S])%mod;
}
printf("%u",1ull*ans*ifac[m]%mod+1);
}
LGP5279题解的更多相关文章
- 2016 华南师大ACM校赛 SCNUCPC 非官方题解
我要举报本次校赛出题人的消极出题!!! 官方题解请戳:http://3.scnuacm2015.sinaapp.com/?p=89(其实就是一堆代码没有题解) A. 树链剖分数据结构板题 题目大意:我 ...
- noip2016十连测题解
以下代码为了阅读方便,省去以下头文件: #include <iostream> #include <stdio.h> #include <math.h> #incl ...
- BZOJ-2561-最小生成树 题解(最小割)
2561: 最小生成树(题解) Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1628 Solved: 786 传送门:http://www.lyd ...
- Codeforces Round #353 (Div. 2) ABCDE 题解 python
Problems # Name A Infinite Sequence standard input/output 1 s, 256 MB x3509 B Restoring P ...
- 哈尔滨理工大学ACM全国邀请赛(网络同步赛)题解
题目链接 提交连接:http://acm-software.hrbust.edu.cn/problemset.php?page=5 1470-1482 只做出来四道比较水的题目,还需要加强中等题的训练 ...
- 2016ACM青岛区域赛题解
A.Relic Discovery_hdu5982 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/65536 K (Jav ...
- poj1399 hoj1037 Direct Visibility 题解 (宽搜)
http://poj.org/problem?id=1399 http://acm.hit.edu.cn/hoj/problem/view?id=1037 题意: 在一个最多200*200的minec ...
- 网络流n题 题解
学会了网络流,就经常闲的没事儿刷网络流--于是乎来一发题解. 1. COGS2093 花园的守护之神 题意:给定一个带权无向图,问至少删除多少条边才能使得s-t最短路的长度变长. 用Dijkstra或 ...
- CF100965C题解..
求方程 \[ \begin{array}\\ \sum_{i=1}^n x_i & \equiv & a_1 \pmod{p} \\ \sum_{i=1}^n x_i^2 & ...
随机推荐
- Docker Harbor私有仓库部署与管理 (超详细配图)
Docker Harbor私有仓库部署与管理 1.Harbor 介绍 2.Harbor部署 3.Harbor管理 1.Harbor 介绍: 什么是 Harbor ? Harbor 是 VMware 公 ...
- Ansible自动化运维工具及其常用模块
Ansible自动化运维工具及其常用模块 目录 Ansible自动化运维工具及其常用模块 一.Ansible简介 1. Ansible概述 2. Ansible作用 3. Ansible的工作模块 4 ...
- Centos7系统使用yum遇到的问题failure: repodata/repomd.xml from base: [Errno 256] No more mirrors to try.
简单粗暴重新安装yum. 1.查看linux上所有的yum包 # rpm -qa|grep yum 2.逐个卸载,如 # rpm -e yum-plugin-fastestmirror-1.1.31- ...
- Vue 组件库:Element
目录 Element 介绍 什么是 Element ? Element 快速入门 Element 常用组件 基础布局 容器布局 表单组件 表格组件 顶部导航栏组件 侧边导航栏组件 Element 介绍 ...
- 08python语法入门--基本数据类型及内置方法
数字类型int与float 定义 类型转换 使用 字符串 定义 类型转换 使用 优先掌握的操作 需要掌握的操作 了解操作 列表 定义 类型转化 使用 优先掌握的操作 需要掌握的操作 了解操作 元组 作 ...
- opencv笔记--SURF
SURF(Speeded-Up Robust Features) 是对 SIFT 得改进,相对于 SIFT,SURF 利用积分图像与盒函数模拟 DoG,提升了计算速度:同时,使用了一种不用于 SIFT ...
- Solution -「LOCAL」画画图
\(\mathcal{Description}\) OurTeam. 给定一棵 \(n\) 个点的树形随机的带边权树,求所有含奇数条边的路径中位数之和.树形生成方式为随机取不连通两点连边直到全 ...
- .net core 和 WPF 开发升讯威在线客服系统:调用有道翻译接口实现实时自动翻译的方法
业余时间用 .net core 写了一个在线客服系统.并在博客园写了一个系列的文章,写介绍这个开发过程. 我把这款业余时间写的小系统丢在网上,陆续有人找我要私有化版本,我都给了,毕竟软件业的初衷就是免 ...
- ios开发 Pods工具心得
Pods 这也是我的第一篇微博,希望能给大家带来帮助,也便于我自己温习 第一步:新建一个xcode项目(这个不解释了) 第二步:打开终端(剩下的操作都在终端里面了)
- linux服务器登录微信报警通知
linux服务器ssh登录安全规则中,理论上要做到:防火墙限制,hosts.allow限制,root禁止登录,用户权限分配,ssh端口修改,登录时间限制,堡垒机登录. 这里只讲登录时间限制和登录报警通 ...