首页 > 其他分享 >23 年牛客提高组模拟赛 Day5 T3

23 年牛客提高组模拟赛 Day5 T3

时间:2023-10-13 10:37:47浏览次数:38  
标签:23 int void Day5 T3 leq read dp define

给你一个长为 \(n\) 的数组 \(b_i\) 表示原数组 \(a_i\) 中以 \(i\) 结尾的 LIS 长度,问对于所有 \(1 \leq a_i \leq m\) ,原数组有多少种不同的可能

\(n \leq 20, m \leq 3000\)

看到数据范围容易想到状压 dp ,赛事想了个比较朴素的 dp :设 \(dp_{S,i}\) 表示填了集合 \(S\) 的数,其中填的最大的数是 \(i\) 的方案数。转移即枚举下一个要填的数是什么,假如枚举的是 \(i\) ,要满足以下条件: \(\max\limits_{j \leq i, j \in S} a_j + 1 = a_i\) ,原因自己想。

发现我们没必要真把数填进去,而是可以确立他们的相对大小关系之后用组合数填数,也就是说我们把要填的数范围变成了 \([1,n]\) 。故设 \(dp_{S,i}\) 表示填了集合 \(S\) 的数,填的最大的数是 \(i\) 的方案数。转移即枚举 \(S\) 子集,满足的条件同理。最终答案即为 \(\sum\limits_{i=1}^{n} dp_{2^n-1, i} \times \binom{m}{i}\) ,复杂度 \(O(3^n \times n^2)\)

发现枚举子集是孬的,因此考虑顺着填答案,具体的,设 \(dp_{i,j,S}\) 表示填了前 \(i\) 种数,第 \(i\) 种考虑到 \(j\) 位置填/不填,填过的集合为 \(S\) 的方案数。容易得到转移:

\[\begin{cases} dp_{i,j,S} \rightarrow dp_{i,j+1,S} & (j < n-1) \\ dp_{i,j,S} \rightarrow dp_{i+1,0,S} & (j = n-1) \\ dp_{i,j,S} \rightarrow dp_{i,j+1,S+j} & (\max_{j \leq i} a_j + 1 = a_i, j < n-1) \\ dp_{i,j,S} \rightarrow dp_{i+1,0,S+j} & (\max_{j \leq i} a_j + 1 = a_i, j = n-1) \\ \end{cases} \]

还有一个问题,怎么计算答案?因为我们并不知道第 \(i\) 个数到底填没填

第一种方法是在 dp 中再记录第 \(i\) 个数填/没填

第二种方法是考虑容斥,设 \(g_i = dp_{i+1,0,2^n-1}\) ,即填 \(\leq i\) 个数的方案数。我们想求 \(f_i\) 表示恰好填 \(i\) 个的方案数。发现满足:

\[f_n = \sum\limits_{i=0}^{n} \binom{n}{i} g_i \]

这是一个显然的子集形式的二项式反演问题,考虑容斥减去没填的方案,即:

\[f_n = \sum\limits_{i=1}^{n} (-1)^{n-i} \binom{n}{i} g_i \]

最终复杂度 \(O(2^n n^2)\)

code :

#include <bits/stdc++.h>
// #pragma GCC optimize(2)
#define pcn putchar('\n')
#define ll long long
#define LL __int128
#define pii pair<int, int>
#define pli pair<ll, int>
#define pil pair<int, ll>
#define pll pair<ll, ll>
#define MP make_pair
#define fi first
#define se second
#define gsize(x) ((int)(x).size())
#define Min(a, b) (a = min(a, b))
#define Max(a, b) (a = max(a, b))
#define For(i, j, k) for(int i = (j), END##i = (k); i <= END##i; ++ i)
#define For__(i, j, k) for(int i = (j), END##i = (k); i >= END##i; -- i)
#define Fore(i, j, k) for(int i = (j); i; i = (k))
using namespace std;

namespace IO {
	template <typename T> T read(T &num){
	    num = 0; T f = 1; char c = ' '; while(c < 48 || c > 57) if((c = getchar()) == '-') f = -1;
	    while(c >= 48 && c <= 57) num = (num << 1) + (num << 3) + (c ^ 48), c = getchar();
	    return num *= f;
	}
	ll read(){
	    ll num = 0, f = 1; char c = ' '; while(c < 48 || c > 57) if((c = getchar()) == '-') f = -1;
	    while(c >= 48 && c <= 57) num = (num << 1) + (num << 3) + (c ^ 48), c = getchar();
	    return num * f;
	}
	template <typename T> void Write(T x){
	    if(x < 0) putchar('-'), x = -x;
		if(x == 0){putchar('0'); return ;}
	    if(x > 9) Write(x / 10);
		putchar('0' + x % 10); return ;
	}
	void putc(string s){ int len = s.size() - 1; For(i, 0, len) putchar(s[ i ]); }
	template <typename T> void write(T x, string s = "\0"){ Write( x ), putc( s ); }
}
using namespace IO;

//mt19937_64 rnd(time(0));
//ll random(ll l, ll r){ return (rnd() % (r - l + 1)) + l; }

#ifdef LOCAL
	template <typename T> void debug(T x, string s = "\0"){ write(x, s); }
#else
	template <typename T> void debug(T x, string s = "\0"){}
#endif

/* ====================================== */

const int maxn = 25;
const int maxm = 3050;
const int maxS = (1 << 20) + 50;
const ll mod = 998244353;

int n, m; int a[ maxn ], maxs[ maxS ];
int C[ maxm ][ maxm ];
int dp[ 2 ][ maxn ][ maxS ]; // dp_{i,j,S} 表示填前 i 种数,第 i 种考虑到 j 位置,填数集合为 S 方案数
// 因为题解是从后往前考虑所以我写的也是
ll ans[ maxn ];

template<typename T> void add(T &x, T y){ x += y; if(x >= mod) x -= mod; } 

void mian(){
	read(n), read(m); For(i, 1, n) read(a[ i ]);
	For(S, 0, (1 << n) - 1) For(i, 0, n - 1) if(S >> i & 1) Max(maxs[ S ], a[ i + 1 ]); // 记录集合 S 中数的最大值
	int o = 0; dp[ o ][ n - 1 ][ 0 ] = 1;
	For(i, 1, n){
		For__(j, n - 1, 0)
			For(S, 0, (1 << n) - 1){
				int &nw = dp[ o ][ j ][ S ]; if(!nw) continue;
				if(j) add(dp[ o ][ j - 1 ][ S ], nw);
				else add(dp[ o ^ 1 ][ n - 1 ][ S ], nw);
				if((S >> j & 1) == 0 && maxs[ S & ((1 << j) - 1) ] + 1 == a[ j + 1 ]){
					if(j) add(dp[ o ][ j - 1 ][ S | (1 << j) ], nw);
					else add(dp[ o ^ 1 ][ n - 1 ][ S | (1 << j) ], nw);
				}
				dp[ o ][ j ][ S ] = 0; // 注意清空
			}
		ans[ i ] = dp[ o ^ 1 ][ n - 1 ][ (1 << n) - 1 ]; o ^= 1;
        // 记录答案,这里如果没有80行清空操作的话写 dp[ o ][ 0 ][ (1 << n) - 1 ] 也是对的
	}
	ll res = 0;
	For(i, 1, n){
		ll t = 0; For(j, 1, i){ t = (t + 1ll * ((i - j & 1) ? mod - 1 : 1) * C[ i ][ j ] % mod * ans[ j ]) % mod; }
		res = (res + t * C[ m ][ i ]) % mod;
	} // 容斥
	write(res, "\n");
	
}

void init(){

}

void treatment(){
	int up = 3005;
	C[ 0 ][ 0 ] = 1; For(i, 1, up){ C[ i ][ 0 ] = 1; For(j, 1, i){ C[ i ][ j ] = C[ i - 1 ][ j ] + C[ i - 1 ][ j - 1 ]; if(C[ i ][ j ] >= mod) C[ i ][ j ] -= mod; } }
}

int main() {

#ifdef LOCAL
	freopen("data.in", "r", stdin);
//	freopen("data.out", "w", stdout);
#else
//	freopen("data.in", "r", stdin);
//	freopen("data.out", "w", stdout);
#endif

	treatment();
	int T = 1;
//	read(T);
	while(T --){
		init();
		mian();
	}

	return 0;
}

标签:23,int,void,Day5,T3,leq,read,dp,define
From: https://www.cnblogs.com/fox-konata/p/17761459.html

相关文章

  • 2023-10-13
    一、初见成效 二、存在问题1.下载口接口:单纯用导线可以直接连接,但是不好看。2.接3.3V电源,led灯亮度不够。3.加了松香的贴片C8T6取不下来。......
  • 2023-10-12 裸k交易法 单根k线 长阴线/长阳线
    长阴线 ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------......
  • 2023-10-12 javac : 无法将“javac”项识别为 cmdlet、函数、脚本文件或可运行程序的
    找到你的jdk安装路径/bin,复制并扔到环境变量中去即可,如:   ......
  • 2023 CSP-J/S 第一轮游记
    Day-1教练说要提前带一点干粮,因为一中没有开食堂啊啊啊啊啊啊啊啊啊啊,要坐校车会学校吃饭,如果路上堵的话就直接在校车上吃了,所以去了趟小卖部买了一袋面包和巧克力,花了快\(30\)元。贵爆了!赶紧倒闭!Day1跟校车(水泥搅拌车)去一中,早上入门组挺简单,但是人真的太多了。阅读程序......
  • 2023-10-12 java学习笔记
    1.安装java环境,点击链接前往下载......
  • 23.10.12
    Java类的继承问题 publicclassParentChildTest{publicstaticvoidmain(String[]args){Parentparent=newParent();parent.printValue();Childchild=newChild();child.printValue();parent=child;parent.printValue()......
  • 2023.10.12
    Redisson如何实现分布式锁。移除链表操作、链表中五个常见的操作get(index):获取链表中第index个节点的值。如果索引无效,则返回-1。addAtHead(val):在链表的第一个元素之前添加一个值为val的节点。插入后,新节点将成为链表的第一个节点。addAtTail(val):将值为val的节点追加......
  • 20231012打卡
    上午,我首先学习了UML(统一建模语言)。老师要求我们完善实验内容,并进行两级功能构建。通过细致的修改和改善,我们的实验内容变得更加完善和规范。UML是软件工程中非常重要的工具,它可以帮助我们进行软件系统的建模和设计,提高开发效率和质量。接着是体育课,我们进行了篮球练习。在练习中......
  • 每日总结20231012
    代码时间(包括上课)8h代码量(行):300行博客数量(篇):1篇相关事项:1、今天是周四,今天上午上的是软件设计和软件需求分析,软件设计讲的是解释器模式和迭代器模式,软件需求分析讲的是如何确定业务范围。2、今天下午上的是人机交互技术,上的是上机课,写的是期末的两个报告和大作业。3、今天还......
  • 2023.10.12——每日总结
    学习所花时间(包括上课):9h代码量(行):0行博客量(篇):1篇今天,上午学习,下午学习;我了解到的知识点:1.软件需求分析;明日计划:学习......