ARC118E Avoid Permutations
题意
给定一个长度为 \(n\) 的排列 \(p\),在一个 \((n + 2)\times(n + 2)\) 的网格上,禁止通过 \((i, p_i)\) 这些点,每次只能向上或右走一格,从 \((0, 0)\) 走到 \((n + 1, n + 1)\) 的方案数,定义为排列的权值。给定一个不完整的排列,对于所有补全排列的方案,计算权值和。
数据范围:\(1\le n \le 200\)。
做法
这题并不需要容斥。
考虑计算对于某一条路径,有多少排列满足没有点在路径上。那么正常 DP 路径,在 \((i, j)\) 这个点向上转移到 \((i, j + 1)\) 的时候决策 \(y = j + 1\) 这一行左侧是否放置一个物品、放置在哪个位置,同时记录当前位置左下角区域总共放置了多少障碍。DP 状态设计为 \(f_{i, j, k, 0/1, 0/1}\) 表示在 \((i, j)\) 左下角放了 \(k\) 个障碍,这一行左边是不是已经放了东西,这一列下面是不是已经放了东西。转移需要一些讨论。时间复杂度 \(\Theta(n^3)\)。
代码
// Author: kyEEcccccc
#include <bits/stdc++.h>
using namespace std;
using LL = long long;
using ULL = unsigned long long;
#define F(i, l, r) for (int i = (l); i <= (r); ++i)
#define FF(i, r, l) for (int i = (r); i >= (l); --i)
#define MAX(a, b) ((a) = max(a, b))
#define MIN(a, b) ((a) = min(a, b))
#define SZ(a) ((int)((a).size()) - 1)
const int N = 205, MOD = 998244353;
int n, p[N], q[N];
int toti[N], totj[N];
LL f[N][N][N][2][2];
signed main(void)
{
	// freopen(".in", "r", stdin);
	// freopen(".out", "w", stdout);
	ios::sync_with_stdio(0), cin.tie(nullptr);
	cin >> n;
	F(i, 1, n)
	{
		cin >> p[i];
		if (p[i] != -1) q[p[i]] = i;
	}
	F(i, 1, n)
	{
		toti[i] = toti[i - 1];
		totj[i] = totj[i - 1];
		if (p[i] <= 0) toti[i] += 1;
		if (q[i] <= 0) totj[i] += 1;
	}
	toti[n + 1] = toti[n];
	totj[n + 1] = totj[n];
	F(i, 0, n + 1) f[i][0][0][0][0] = f[0][i][0][0][0] = 1;
	F(i, 1, n + 1) F(j, 1, n + 1) F(k, 0, min(toti[i], totj[j])) F(a, 0, 1) F(b, 0, 1)
	{
		LL &res = f[i][j][k][a][b];
		if (((a == 1 && p[i] <= 0) || (b == 1 && q[j] <= 0)) && k == 0) continue;
		if ((i == n + 1 && a == 1) || (j == n + 1 && b == 1)) continue;
		if (p[i] > 0 && p[i] == j) continue;
		if (p[i] > 0 && ((p[i] < j && a == 0) || (p[i] > j && a == 1))) continue;
		if (q[j] > 0 && ((q[j] < i && b == 0) || (q[j] > i && b == 1))) continue;
		if (a == 0)
		{
			res = (res + f[i - 1][j][k][0][b] + f[i - 1][j][k][1][b]) % MOD;
		}
		else
		{
			if (p[i] > 0)
			{
				res = (res + f[i - 1][j][k][0][b] + f[i - 1][j][k][1][b]) % MOD;
			}
			else if (q[j] <= 0)
			{
				res = (res + (f[i - 1][j][k - 1][0][b] + f[i - 1][j][k - 1][1][b])
					* (totj[j - 1] - k + 1 + b)) % MOD;
			}
			else
			{
				res = (res + (f[i - 1][j][k - 1][0][b] + f[i - 1][j][k - 1][1][b])
					* (totj[j - 1] - k + 1)) % MOD;
			}
		}
		if (b == 0)
		{
			res = (res + f[i][j - 1][k][a][0] + f[i][j - 1][k][a][1]) % MOD;
		}
		else
		{
			if (q[j] > 0)
			{
				res = (res + f[i][j - 1][k][a][0] + f[i][j - 1][k][a][1]) % MOD;
			}
			else if (p[i] <= 0)
			{
				res = (res + (f[i][j - 1][k - 1][a][0] + f[i][j - 1][k - 1][a][1])
					* (toti[i - 1] - k + 1 + a)) % MOD;
			}
			else
			{
				res = (res + (f[i][j - 1][k - 1][a][0] + f[i][j - 1][k - 1][a][1])
					* (toti[i - 1] - k + 1)) % MOD;
			}
		}
	}
	cout << f[n + 1][n + 1][toti[n]][0][0] << endl;
	return 0;
}
ARC118E Avoid Permutations的更多相关文章
- 47. Permutations II (JAVA)
		Given a collection of numbers that might contain duplicates, return all possible unique permutations ... 
- Permutations II
		Given a collection of numbers that might contain duplicates, return all possible unique permutations ... 
- [LeetCode] Permutations II 全排列之二
		Given a collection of numbers that might contain duplicates, return all possible unique permutations ... 
- [LeetCode] Permutations 全排列
		Given a collection of numbers, return all possible permutations. For example,[1,2,3] have the follow ... 
- POJ2369 Permutations(置换的周期)
		链接:http://poj.org/problem?id=2369 Permutations Time Limit: 1000MS Memory Limit: 65536K Total Submi ... 
- String源码中的"avoid getfield opcode"
		引言: 之前一篇文章梳理了String的不变性原则,还提到了一段源码中注释"avoid getfield opcode",当时通过查阅资料发现,这是为了防止 getfield(获取 ... 
- 10 Biggest Business Mistakes That Every Entrepreneur Should Avoid
		原文链接:http://www.huffingtonpost.com/syed-balkhi/10-biggest-business-mista_b_7626978.html When I start ... 
- Permutations
		Permutations Given a collection of distinct numbers, return all possible permutations. For example,[ ... 
- 【leetcode】Permutations
		题目描述: Given a collection of numbers, return all possible permutations. For example, [1,2,3] have the ... 
- [leetcode] 47. Permutations II
		Given a collection of numbers that might contain duplicates, return all possible unique permutations ... 
随机推荐
- Nmap基本使用【速查版】
			列举远程机器开放的端口 nmap [域名] 列举远程机器开放的端口和服务 nmap --dns-servers 8.8.8.8 [域名] nmap进行探测之前要把域名通过DNS服务器解析为IP地址,我 ... 
- Troubleshooting 专题 - 问正确的问题 得到正确的答案
			在很多公司中,IT.数据中心.业务系统一出故障,会有很多人被叫到作战室(就是一个为了解决该问题,而把所有相关人员集中在一起的一个会议室), 但是对于这个问题他们是否可以修复, 是否他们应该负有责任, ... 
- .net6的IIS发布部署
			1.打开控制面板,打开程序 2.点击启动或关闭windows功能 3.在其中选择要设置的IIS功能 4.重启IIS服务 5.发布项目 6.在开始菜单搜索IIS,点击IIS管理器 7.右击网站,点击添加 ... 
- k8s 1.24 1.25 集群使用docker作为容器
			Kubernetes 1.24 1.25 集群使用docker作为容器 背景 在新版本Kubernetes环境(1.24以及以上版本)下官方不在支持docker作为容器运行时了,若要继续使用docke ... 
- flask-login使用方法
			烧瓶登录 Flask-Login 为 Flask 提供用户会话管理.它处理登录.注销和长时间记住用户会话的常见任务. 它会: 将活动用户的 ID 存储在Flask Session中,让您轻松登录和注销 ... 
- spring-boot-starter-parent not found
			项目开始的springboot版本为2.6.6 <parent> <groupId>org.springframework.boot</groupId> <a ... 
- 迁移学习()《Attract, Perturb, and Explore: Learning a Feature Alignment Network for Semi-supervised Domain Adaptation》
			论文信息 论文标题:Attract, Perturb, and Explore: Learning a Feature Alignment Network for Semi-supervised Do ... 
- day25:7个魔术方法&5个关于类的魔术属性
			目录 1.__del__(析构方法) 2.魔术方法:__str__ 3.魔术方法:__repr__ 4.魔术方法:__call__ 5.魔术方法:__bool__ 6.魔术方法:__add__& ... 
- Docker中Nginx搭建以及配置
			docker nginx搭建 1 docker pull nginx docker pull nginx 2 启动nginx docker run --name nginx -p 80:80 -d n ... 
- Java中的同步
			Java中的同步 线程间的通讯首要的方式就是对字段及其字段所引用的对象的共享访问.这种通信方式是及其高效的,但是也是导致了可能的错误:线程间相互干涉和内存一致性的问题.避免出现这两种错误的方法就是 ... 
