【LOJ】#3042. 「ZJOI2019」麻将
LOJ#3042. 「ZJOI2019」麻将
如何判定一个集合牌有没有胡的子集是不是胡的
就用一个\(dp[j][k][0/1]\)表示有j个连续两个的串,有k个连续1个串,有没有对子,再记一下这个集合里的牌大于等于2的花色数有几个
我们把\(dp[j][k][0/1]\)和大于等于2的花色数作为一副牌的状态,然后给每个状态标号,做一个dp
\(f[i][j][S]\)表示考虑到第\(i\)种花色,有\(j\)张牌,状态标号是\(S\)的方案数,记录到第\(j\)张牌还没赢的方案数是\(S(j)\),那么期望就是\(\frac{\sum_{j = 13}^{\infty} S(j)}{(4n- 13)!}\)
#include <bits/stdc++.h>
#define fi first
#define se second
#define pii pair<int,int>
#define mp make_pair
#define pb push_back
#define space putchar(' ')
#define enter putchar('\n')
#define eps 1e-10
#define MAXN 200005
#define ba 47
//#define ivorysi
using namespace std;
typedef long long int64;
typedef unsigned int u32;
typedef double db;
template<class T>
void read(T &res) {
res = 0;T f = 1;char c = getchar();
while(c < '0' || c > '9') {
if(c == '-') f = -1;
c = getchar();
}
while(c >= '0' && c <= '9') {
res = res * 10 +c - '0';
c = getchar();
}
res *= f;
}
template<class T>
void out(T x) {
if(x < 0) {x = -x;putchar('-');}
if(x >= 10) {
out(x / 10);
}
putchar('0' + x % 10);
}
const int MOD = 998244353;
int inc(int a,int b) {
return a + b >= MOD ? a + b - MOD : a + b;
}
int mul(int a,int b) {
return 1LL * a * b % MOD;
}
void update(int &x,int y) {
x = inc(x,y);
}
void upmax(int &x,int y) {
x = max(x,min(4,y));
}
int fpow(int x,int c) {
int res = 1,t = x;
while(c) {
if(c & 1) res = mul(res,t);
t = mul(t,t);
c >>= 1;
}
return res;
}
int N;
int cnt[105];
struct state {
int d[3][3];
state() {memset(d,-1,sizeof(d));}
friend state trans(const state &a,int t) {
state c;
for(int j = 0 ; j <= 2 ; ++j) {
for(int k = 0 ; k <= 2 ; ++k) {
if(a.d[j][k] == -1) continue;
for(int i = 0 ; i <= min(2,t - j - k) ; ++i) {
upmax(c.d[k][i],a.d[j][k] + j + (t - j - k - i) / 3);
}
}
}
return c;
}
friend state Max(const state &a,const state &b) {
state t;
for(int j = 0 ; j <= 2 ; ++j) {
for(int k = 0 ; k <= 2 ; ++k) {
t.d[j][k] = max(a.d[j][k],b.d[j][k]);
}
}
return t;
}
friend bool operator == (const state &a,const state &b) {
for(int j = 0 ; j <= 2 ; ++j) {
for(int k = 0 ; k <= 2 ; ++k) {
if(a.d[j][k] != b.d[j][k]) return false;
}
}
return true;
}
friend bool operator != (const state &a,const state &b) {
return !(a == b);
}
friend bool operator < (const state &a,const state &b) {
for(int j = 0 ; j <= 2 ; ++j) {
for(int k = 0 ; k <= 2 ; ++k) {
if(a.d[j][k] != b.d[j][k]) return a.d[j][k] < b.d[j][k];
}
}
return false;
}
};
struct MJ {
state dp[2];int cnt;
MJ() {cnt = 0;dp[0].d[0][0] = 0;}
friend MJ Trans(const MJ &a,int t) {
MJ c;
c.dp[1] = trans(a.dp[1],t);
if(t >= 2) c.dp[1] = Max(trans(a.dp[0],t - 2),c.dp[1]);
c.dp[0] = trans(a.dp[0],t);
if(t >= 2) c.cnt = min(a.cnt + 1,7);
else c.cnt = a.cnt;
return c;
}
friend bool operator == (const MJ &a,const MJ &b) {
for(int i = 0 ; i < 2 ; ++i) if(a.dp[i] != b.dp[i]) return false;
if(a.cnt != b.cnt) return false;
return true;
}
friend bool operator < (const MJ &a,const MJ &b) {
for(int i = 0 ; i < 2 ; ++i) if(a.dp[i] != b.dp[i]) return a.dp[i] < b.dp[i];
if(a.cnt != b.cnt) return a.cnt < b.cnt;
return false;
}
bool win() {
if(cnt >= 7) return true;
for(int i = 0 ; i <= 2 ; ++i) {
for(int j = 0 ; j <= 2 ; ++j) {
if(dp[1].d[i][j] >= 4) return true;
}
}
return false;
}
void print() {
out(cnt);enter;
for(int p = 0 ; p <= 1 ; ++p) {
for(int j = 0 ; j <= 2 ; ++j) {
for(int k = 0 ; k <= 2 ; ++k) {
out(dp[p].d[j][k]);space;
}
enter;
}
}
}
}pool[100006];
map<MJ,int> zz;
int tot;
int dp[2][405][5005],cur,fac[405],invfac[405];
void dfs(MJ a) {
if(zz.count(a)) return;
if(a.win()) return;
zz[a] = ++tot;pool[tot] = a;
for(int i = 0 ; i <= 4 ; ++i) {
dfs(Trans(a,i));
}
}
int C(int n,int m) {
if(n < m) return 0;
return mul(fac[n],mul(invfac[m],invfac[n - m]));
}
void Solve() {
read(N);
int w,a;
for(int i = 1 ; i <= 13 ; ++i) {
read(w);read(a);
cnt[w]++;
}
MJ t;
dfs(t);
dp[0][0][zz[t]] = 1;cur = 0;
fac[0] = 1;
for(int i = 1 ; i <= 400 ; ++i) fac[i] = mul(fac[i - 1],i);
invfac[400] = fpow(fac[400],MOD - 2);
for(int i = 399 ; i >= 0 ; --i) invfac[i] = mul(invfac[i + 1],i + 1);
for(int i = 1 ; i <= N ; ++i) {
memset(dp[cur ^ 1],0,sizeof(dp[cur ^ 1]));
for(int j = 0 ; j <= 4 * (i - 1) ; ++j) {
for(int h = 1 ; h <= tot ; ++h) {
if(!dp[cur][j][h]) continue;
for(int k = max(cnt[i],0) ; k <= 4 ; ++k) {
MJ nxt = Trans(pool[h],k);
if(nxt.win()) continue;
update(dp[cur ^ 1][j + k][zz[nxt]],mul(dp[cur][j][h],C(4 - cnt[i],k - cnt[i])));
}
}
}
cur ^= 1;
}
int ans = 0;
for(int j = 13 ; j <= 4 * N ; ++j) {
for(int h = 1 ; h <= tot ; ++h) {
update(ans,mul(dp[cur][j][h],mul(fac[j - 13],fac[4 * N - j])));
}
}
ans = mul(ans,invfac[4 * N - 13]);
out(ans);enter;
}
int main() {
#ifdef ivorysi
freopen("f1.in","r",stdin);
#endif
Solve();
return 0;
}
【LOJ】#3042. 「ZJOI2019」麻将的更多相关文章
- Loj #3042. 「ZJOI2019」麻将
Loj #3042. 「ZJOI2019」麻将 题目描述 九条可怜是一个热爱打麻将的女孩子.因此她出了一道和麻将相关的题目,希望这题不会让你对麻将的热爱消失殆尽. 今天,可怜想要打麻将,但是她的朋友们 ...
- Loj #3044. 「ZJOI2019」Minimax 搜索
Loj #3044. 「ZJOI2019」Minimax 搜索 题目描述 九条可怜是一个喜欢玩游戏的女孩子.为了增强自己的游戏水平,她想要用理论的武器武装自己.这道题和著名的 Minimax 搜索有关 ...
- Loj #3045. 「ZJOI2019」开关
Loj #3045. 「ZJOI2019」开关 题目描述 九条可怜是一个贪玩的女孩子. 这天,她和她的好朋友法海哥哥去玩密室逃脱.在他们面前的是 \(n\) 个开关,开始每个开关都是关闭的状态.要通过 ...
- 【线段树 树链剖分 差分 经典技巧】loj#3046. 「ZJOI2019」语言【未完】
还是来致敬一下那过往吧 题目分析 先丢代码 #include<bits/stdc++.h> ; ; ; struct node { int top,son,fa,tot; }a[maxn] ...
- 「ZJOI2019」麻将
传送门 Solution 对于条件一:记录一个\(cnt\)表示牌个数\(≥2\)的个数 设\(dp_{i,0/1,j,k}\)表示考虑了\(1...i\),当前是否有对子,以\(i-1\),\(i ...
- @loj - 3043@「ZJOI2019」线段树
目录 @description@ @solution@ @accepted code@ @details@ @description@ 九条可怜是一个喜欢数据结构的女孩子,在常见的数据结构中,可怜最喜 ...
- @loj - 3046@「ZJOI2019」语言
目录 @description@ @solution@ @accepted code@ @details@ @description@ 九条可怜是一个喜欢规律的女孩子.按照规律,第二题应该是一道和数据 ...
- 【LOJ】#3046. 「ZJOI2019」语言
LOJ#3046. 「ZJOI2019」语言 先orz zsy吧 有一个\(n\log^3n\)的做法是把树链剖分后,形成logn个区间,这些区间两两搭配可以获得一个矩形,求矩形面积并 然后就是对于一 ...
- 【LOJ】#3044. 「ZJOI2019」Minimax 搜索
LOJ#3044. 「ZJOI2019」Minimax 搜索 一个菜鸡的50pts暴力 设\(dp[u][j]\)表示\(u\)用\(j\)次操作能使得\(u\)的大小改变的方案数 设每个点的初始答案 ...
随机推荐
- [Luogu] 次小生成树
https://www.luogu.org/problemnew/show/P4180#sub 严格次小生成树,即不等于最小生成树中的边权之和最小的生成树 首先求出最小生成树,然后枚举所有不在最小生成 ...
- 【csp模拟赛2】黑莲花--数据结构+数论
没有什么能够阻挡,你对被阿的向往.天天 AK 的生涯,你的心了无牵挂. 虐过大佬的比赛,也曾装弱装逼.当你低头的瞬间,才发现旁边的人. 把你的四肢抬起来,使劲地往门上撞.盛开着永不凋零,黑莲花. —— ...
- POJ - 3376 Finding Palindromes manacher+字典树
题意 给n个字符串,两两拼接,问拼接后的\(n\times n\)个字符串中有多少个回文串. 分析 将所有正串插入字典树中,马拉车跑出所有串哪些前缀和后缀为回文串,记录位置,用反串去字典树中查询,两字 ...
- 如何检测域名是否被微信屏蔽 微信域名检测接口API是如何实现
微信域名检测技术的主要用户是微信域名防封,大家知道拼多多这种网站,靠诱导分享方式在微信里面摇身一变已经估值160亿美元,身价仅次于京东了 ,这是何等的速度,简直是惊为天人,but 如果你想玩微信病毒营 ...
- MySort(选做)
一.题目要求 注意:研究sort的其他功能,要能改的动代码,需要答辩 模拟实现Linux下Sort -t : -k 2的功能. 要有伪代码,产品代码,测试代码(注意测试用例的设计) 参考 Sort的实 ...
- Ubuntu使用PBIS认证
1:下载 https://github.com/BeyondTrust/pbis-open/releases wget https://github.com/BeyondTrust/pbis-open ...
- python 文件压缩及解压
文件压缩 import os import zipfile def zip_dir(dirname,zipfilename): """ | ##@函数目的: 压缩指定目录 ...
- tkinter入门-布局方式pack(), grid(), place()
pack 转载于https://www.cnblogs.com/kongzhagen/p/6144588.html\ 1. 使用pack函数的时候,默认先使用放到上面的,然后依次从上向下排 2. 可接 ...
- npm 权限
将npm默认目录定向到其他你具有读写权限的目录 很多时候你可能并不想改变npm所用的默认目录(如/usr)的拥有者,因为这可能会导致一些问题,比如你在与其他用户共用此系统时. 这时,你可以设置npm整 ...
- Win10蓝牙鼠标老是断连卡顿的解决方法
一直用一个微软家的蓝牙鼠标,饱受鼠标卡顿困扰,今天找到了一个解决方案,用了下,效果显著.具体操作见下文. 原文地址:https://jingyan.baidu.com/article/c85b7a64 ...