【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/P3130 #include <cstdio> #include <iostream> using ...
- vue-cli3项目首页加载速度优化(cdn加速,路由懒加载,gzip压缩)
今天打算上线vue的单页面项目,上线后,首页加载速度巨慢! 原因是项目上线后,网速不够快,加载js,css等资源很慢, 打开打包好的文件发现chunk-vendors.xxxxxxx.js的包很大,达 ...
- 在iOS开发中使用icon font的方法
http://iconfont.cn/help/iconuse.html 在开发阿里数据iOS版客户端的时候,由于项目进度很紧,项目里的所有图标都是用最平常的背景图片方案来实现.而为了要兼容普通屏与R ...
- Oracle中根据列名找到所属的表
oracle中如何根据一个字段名查找出所属的表名? 用如下语句, select * from user_tab_columns where column_name='列名', 例子:select * ...
- PHPStorm2017去掉函数参数提示
今天升级到 PHPStorm 2017.1 发现增加了好些新功能, 有个默认开启的参数名和类型提示功能, 虽然功能挺强大的, 不过我用不着, 还是关掉的好, 有同样需求的同学可以看看 例子比较特殊这么 ...
- Android跨进程通信Content Provider
Content Provider ContentProvider在android中的作用是对外共享数据,也就是说你可以通过ContentProvider把应用中的数据共享给其他应用访问,其他应用可以通 ...
- 在Eclipse中使用WindowBuilder设计Swing程序
在Eclipse中使用WindowBuilder设计Swing程序 Swing程序表示Java的客户端窗体程序,除了通过手动编写代码的方式设计Swing程序之外,Eclipse中还提供了一种W ...
- ThreadLocal详解【使用场景】
转: 么是ThreadLocal 根据JDK文档中的解释:ThreadLocal的作用是提供线程内的局部变量,这种变量在多线程环境下访问时能够保证各个线程里变量的独立性. 从这里可以看出,引入Thre ...
- java.lang.reflect.Method.getAnnotation()方法示例
转: java.lang.reflect.Method.getAnnotation()方法示例 作者: 初生不惑 Java技术QQ群:227270512 / Linux QQ群:479429477 ...
- 几句java代码搞定十万个为什么数据
最近想做一个app,为小朋友推荐十万个为什么的故事,但是找了很久数据,发现挺难的找的,又去写爬虫,发现没有一个好的网站可以爬,最后一个偶然的机会在csdn上发现一个可用的接口,很快就把问题解决了,下面 ...