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」麻将的更多相关文章

  1. Loj #3042. 「ZJOI2019」麻将

    Loj #3042. 「ZJOI2019」麻将 题目描述 九条可怜是一个热爱打麻将的女孩子.因此她出了一道和麻将相关的题目,希望这题不会让你对麻将的热爱消失殆尽. 今天,可怜想要打麻将,但是她的朋友们 ...

  2. Loj #3044. 「ZJOI2019」Minimax 搜索

    Loj #3044. 「ZJOI2019」Minimax 搜索 题目描述 九条可怜是一个喜欢玩游戏的女孩子.为了增强自己的游戏水平,她想要用理论的武器武装自己.这道题和著名的 Minimax 搜索有关 ...

  3. Loj #3045. 「ZJOI2019」开关

    Loj #3045. 「ZJOI2019」开关 题目描述 九条可怜是一个贪玩的女孩子. 这天,她和她的好朋友法海哥哥去玩密室逃脱.在他们面前的是 \(n\) 个开关,开始每个开关都是关闭的状态.要通过 ...

  4. 【线段树 树链剖分 差分 经典技巧】loj#3046. 「ZJOI2019」语言【未完】

    还是来致敬一下那过往吧 题目分析 先丢代码 #include<bits/stdc++.h> ; ; ; struct node { int top,son,fa,tot; }a[maxn] ...

  5. 「ZJOI2019」麻将

    传送门 Solution  对于条件一:记录一个\(cnt\)表示牌个数\(≥2\)的个数 设\(dp_{i,0/1,j,k}\)表示考虑了\(1...i\),当前是否有对子,以\(i-1\),\(i ...

  6. @loj - 3043@「ZJOI2019」线段树

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 九条可怜是一个喜欢数据结构的女孩子,在常见的数据结构中,可怜最喜 ...

  7. @loj - 3046@「ZJOI2019」语言

    目录 @description@ @solution@ @accepted code@ @details@ @description@ 九条可怜是一个喜欢规律的女孩子.按照规律,第二题应该是一道和数据 ...

  8. 【LOJ】#3046. 「ZJOI2019」语言

    LOJ#3046. 「ZJOI2019」语言 先orz zsy吧 有一个\(n\log^3n\)的做法是把树链剖分后,形成logn个区间,这些区间两两搭配可以获得一个矩形,求矩形面积并 然后就是对于一 ...

  9. 【LOJ】#3044. 「ZJOI2019」Minimax 搜索

    LOJ#3044. 「ZJOI2019」Minimax 搜索 一个菜鸡的50pts暴力 设\(dp[u][j]\)表示\(u\)用\(j\)次操作能使得\(u\)的大小改变的方案数 设每个点的初始答案 ...

随机推荐

  1. [Luogu] 花神游历各国

    https://www.luogu.org/problemnew/show/P4145 线段树区间求和 + 区间开根号 对1e9的数开根号下取整用不了几次就会<=1 因此暴力开根号,记录区间最大 ...

  2. java中的变量和数据类型

    变量和javascript的变量含义一样 在Java中,变量分为两种:基本类型的变量和引用类型的变量.(javascript中同样是这样的) 基本数据类型 基本数据类型是CPU可以直接进行运算的类型. ...

  3. JavaWeb_(Spring框架)在Struts+Hibernate框架中引入Spring框架

    spring的功能:简单来说就是帮我们new对象,什么时候new对象好,什么时候销毁对象. 在MySQL中添加spring数据库,添加user表,并添加一条用户数据 使用struts + hibern ...

  4. ubuntu 安装eclipse for c++

    linux的GUI和windos比起来实在逊色,虽然它的终端模式(命令行模式)非常强大.linux发行版ubuntu的GUI相对其他版本要华丽一些,所以最近由redhat转向ubuntu进行linux ...

  5. antd源码分析之——对话框(modal)

    目录 一.组件结构 1.antd代码结构 2.rc-ant代码结构 3.组件结构 二.antd组件调用关系及功能详解 1.Model.tsx 2.confirm 三.rc-dialog详解 1.e.t ...

  6. 如何解决Sublime text中文乱码问题

    Sublime Text 2是一个非常不错的源代码及文本编辑器,但是不支持GB2312和GBK编码在很多情况下会非常麻烦.不过Sublime Package Control所以供的插件可以让Subli ...

  7. Go项目的测试代码1(基础)

    最近写了测试代码,整理了一下. 先看看简单的测试代码. // add_test.go ==> 文件名 _test.go 结尾的默认为测试代码文件 package models import ( ...

  8. 小D课堂-SpringBoot 2.x微信支付在线教育网站项目实战_1-1.SpringBoot整合微信支付开发在线教育视频站点介绍

    笔记 第一章项目介绍和前期准备 1.SpringBoot整合微信支付开发在线教育视频站点介绍     简介: 课程介绍,和小D课堂在线教育项目搭建开发 1.课程大纲介绍         2.微信支付项 ...

  9. EF中where中日期帅选问题

    结合 SqlFunctions  和 DbFunctions  不能同时用两个DbFunctions queryAble = queryAble.Where(s=> SqlFunctions.D ...

  10. java数据结构之WeakHashMap

    一.JAVA中的四种引用类型 1.强引用(StrongReference):强引用是最为普遍的一种引用,如果对象被强引用,那么垃圾回收器无论如何都不会回收它,当内存不足时会抛出OutOfMemoryE ...