【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\)的大小改变的方案数 设每个点的初始答案 ...
随机推荐
- 收藏一个RMQ模板
int a[1100]; int dp[maxn][20]; void rmq_init(){ for(int i=0;i<n;i++) dp[i][0]=a[i]; for(int j=1;( ...
- HGOI20191115 模拟赛 题解
Problem A 表演 有$n$个有点权的点,$m$个有边权的边.对于每个点$u$,输出从这个点出发到$v$,其路径权值的两倍加上v的点权和最小的值. 对于$100\%$的数据,满足$1 \leq ...
- 修改Ubuntu国内镜像
打开软件和更新:如下图 选择一个自己喜欢的镜像. 然后执行 sudo apt-get update 更新源. 结束 1.原文件备份 sudo cp /etc/apt/sources.list ...
- codeforces#1166F. Vicky's Delivery (Service并查集+启发式合并)
题目链接: https://codeforces.com/contest/1166/problem/F 题意: 给出节点数为$n$,边数为$m$的图,保证每个点对都是互连的 定义彩虹路:这条路经过$k ...
- Java中基本数据类型
在数据类型中,最常用也是最基础的数据类型,被称作基本数据类型.可以使用这些类型的值来代表一些简单的状态. Java 语言的基本数据类型总共有以下8 种,下面是按照用途划分出的4 个类别: 定点类型: ...
- LVS之ipvsadm命令
目录: 安装 基本描述 用法 命令选项 示例 [安装] 可使用yum安装或者从官网下载安装包源码安装,两种方式皆可 先检查是否已经安装ipvsadm [root@v_machine1 ~]# yum ...
- 性能优化 | JVM与性能优化知识点综合整理
JVM JVM是java的核心和基础,在java编译器和os平台之间的虚拟处理器.它是一种利用软件方法实现的抽象的计算机基于下层的操作系统和硬件平台,可以在上面执行java的字节码程序. java编译 ...
- P2456 [SDOI2006]二进制方程
P2456 [SDOI2006]二进制方程 题解 拿个样例模拟一下发现 把等式两边对应展开,每个位置的填数都是一一对应的 比如第二个样例 分类讨论: (1)xi yi 都是数字,但是不相同,此时无 ...
- buildscript和allprojects的作用和区别是什么?
在Android Studio的Project的build.gradle中, // Top-level build file where you can add configuration optio ...
- 如何解决AndroidStudio下载资源慢的问题
https://blog.csdn.net/LiangJianxiong/article/details/87881150方法1.build.gradle里的buildscript和allprojec ...