题目链接

题目

题目描述

曾经有一道叫做迷雾森林的题目,然而牛牛认为地图中的障碍太多,实在是太难了,所以删去了很多点,出了这道题。

牛牛给出了一个n行m列的网格图

初始牛牛处在最左下角的格点上(n+1,1),终点在右上角的格点(1,m+1)

现在它想知道,从起点走到终点,只能向上或向右走,一共有多少种走法呢?

需要注意的是,除了起点和终点外,其它的每个格点都有可能有障碍,无法通过。

请注意格子与格点的区别

输入描述

第一行三个整数n,m,k,表示格子的大小,以及有k个障碍

接下来k行(若k=0则无此行),每行一个格点坐标(x,y),表示每个障碍的位置

输出描述

仅一行,一个整数表示答案对998244353取模的值

示例1

输入

4 5 1
3 4

输出

66

说明

示例2

输入

12345 54321 2
123 321
456 654

输出

801071140

示例3

输入

114 514 3
19 19
8 10
65 45

输出

567102428

备注

题解

知识点:计数dp,容斥原理,DAG上dp。

按数据点分治,先将 \(x\) 变换为 \(n+1-x\) ,这样起点就是 \((1,1)\) ,终点就是 \((n+1,m+1)\) 。

当 \(n,m \leq 1000\) 时,就是朴素的路径计数dp,不多讲了。

当 \(n,m > 1000\) 时,注意到 \(k\) 很小,可以考虑容斥原理,求经过至少一个障碍物的路径总数,再用总路径数减去即可,复杂度是 \(O(k \cdot 2^k)\) 的。其中,从 \((x_1,y_1)\) 到 \((x_2,y_2)\) 的全部路径总数为 \(\dbinom{x_2 - x_1 + y_2 - y_1}{x_2 - x_1}\) 。

但我们发现,这里的容斥实际上大部分状态都是不合法的,因为要保证子集的障碍物满足二维偏序,才是一条合法的障碍物路径。然而,我们将障碍物满足的二维偏序图画出来,呈现的一定是个DAG,于是我们可以仿照之前的路径计数dp,再这个DAG上dp。

我们设 \(f_i\) 表示到第 \(i\) 个障碍物但不经过其他障碍物的路径总数。那么 \(f_i\) 等于到达它全部路径总数,减去能到达它的障碍物 \(j\) 的 \(f_j\) 即可。

为了保证在求 \(f_i\) 之前,所有能到达 \(i\) 的障碍物的答案都已经求完了,我们可以建一个DAG图进行拓扑序的dp,但实际上不需要这么做,我们可以仿照二维偏序(逆序对)的方法进行维护。

我们将障碍物按 \(x\) 坐标从小到大排序,这时候就维护了第一维的偏序。之后当我们枚举到 \(i\) ,我们只需要枚举 \(j\) 满足 \(j<i\) 的障碍物即可先保证了第一维的偏序,再检验第二维偏序即可。

最后,我们用总路径数减去所有障碍物的 \(f\) 即可。

时间复杂度 \(O(min\{nm,k^2\})\)

空间复杂度 \(O(nm+k)\)

代码

#include <bits/stdc++.h>
using namespace std;
using ll = long long; const int P = 998244353;
namespace Number_Theory {
const int N = 2e5 + 7;
int qpow(int a, ll k) {
int ans = 1;
while (k) {
if (k & 1) ans = 1LL * ans * a % P;
k >>= 1;
a = 1LL * a * a % P;
}
return ans;
}
int fact[N], invfact[N];
void init(int n) {
fact[0] = 1;
for (int i = 1;i <= n;i++) fact[i] = 1LL * i * fact[i - 1] % P;
invfact[n] = qpow(fact[n], P - 2);
for (int i = n;i >= 1;i--) invfact[i - 1] = 1LL * invfact[i] * i % P;
}
}
namespace CNM {
using namespace Number_Theory;
int C(int n, int m) {
if (n == m && m == -1) return 1; //* 隔板法特判
if (n < m || m < 0) return 0;
return 1LL * fact[n] * invfact[n - m] % P * invfact[m] % P;
}
}
/// 公式法求组合数,O(n),预处理阶乘及其逆元快速求出组合数 int calc(int x1, int y1, int x2, int y2) {
return CNM::C(x2 - x1 + y2 - y1, x2 - x1);
} pair<int, int> obst[100007];
bool dt[1007][1007];
int f[1007][1007];
int ans[100007];
int main() {
std::ios::sync_with_stdio(0), cin.tie(0), cout.tie(0);
int n, m, k;
cin >> n >> m >> k;
for (int i = 1;i <= k;i++) {
int x, y;
cin >> x >> y;
obst[i] = { n + 1 - x + 1,y };
}
if (n <= 1000 && m <= 1000) {
for (int i = 1;i <= k;i++) {
auto [x, y] = obst[i];
dt[x][y] = 1;
}
f[0][1] = 1;
for (int i = 1;i <= n + 1;i++)
for (int j = 1;j <= m + 1;j++)
if (!dt[i][j]) f[i][j] = (f[i - 1][j] + f[i][j - 1]) % P;
cout << f[n + 1][m + 1] << '\n';
}
else {
Number_Theory::init(n + m + 2);
sort(obst + 1, obst + k + 1);
obst[++k] = { n + 1,m + 1 };
for (int i = 1;i <= k;i++) {
auto [x1, y1] = obst[i];
ans[i] = calc(1, 1, x1, y1);
for (int j = 1;j <= i - 1;j++) {
auto [x2, y2] = obst[j];
if (x2 > x1 || y2 > y1) continue;
(ans[i] -= 1LL * ans[j] * calc(x2, y2, x1, y1) % P - P) %= P;
}
}
cout << ans[k] << '\n';
}
return 0;
}

NC200324 魔改森林的更多相关文章

  1. 魔改——MDI多视图模板Tab/标签页 初始化/操作控件

    ==================================声明================================== 本文原创,转载在正文中显要的注明作者和出处,并保证文章的完 ...

  2. 魔改——MFC SDI程序 转换为 MDI程序

    ==================================声明================================== 本文原创,转载在正文中显要的注明作者和出处,并保证文章的完 ...

  3. 魔改——MFC MDI程序 定制 文档模板 运行时全部打开 禁用关闭按钮

    ==================================声明================================== 本文原创,转载在正文中显要的注明作者和出处,并保证文章的完 ...

  4. 【题解】BZOJ4241: 历史研究(魔改莫队)

    [题解]BZOJ4241: 历史研究(魔改莫队) 真的是好题啊 题意 给你一个序列和很多组询问(可以离线),问你这个区间中\(\max\){元素出现个数\(\times\)元素权值} IOI国历史研究 ...

  5. Asp.Net Core Identity 骚断腿的究极魔改实体类

    前言 默认的 Identity 实体类型在大多数时候已经基本够用,很多时候也只是稍微在 IdentityUser 类中增加一些自定义数据字段,比如头像.这次,我要向园友隆重介绍我魔改之后的 Ident ...

  6. 你没有见过的【高恪】船新版本(SX3000 NAT1 X86魔改)

    最近魔改了高恪SX3000 X86,做了如下更改: 开启了SSH 集成了插件(酸酸乳.V2RXY.SMB等等) 开启了NAT1 DIY了主题 精简了官方内置的无用应用和模块 截图(建议右击图片,在新标 ...

  7. [7b2美化]柒比贰 魔改系列|7B2-分类封面添加波浪效果&每日诗词

    本文转载自:钻芒博客 https://www.zmki.cn/5105.html 效果如图: 代码: 首先在style.css样式表里添加波浪样式 /*浪来了*/ .lang { overflow: ...

  8. 【题解】LOJ2462完美的集合(树DP 魔改Lucas)

    [题解]LOJ2462完美的集合(树DP 魔改Lucas) 省选模拟考这个??????????????????? 题目大意: 有一棵树,每个点有两个属性,一个是重量\(w_i\)一个是价值\(v_i\ ...

  9. layui 魔改:富文本编辑器添加上传视频功能

    甲方又整新需求了:富文本编辑器需要可以传视频. layui本身的富文本编辑器没有传视频的功能,所以,又到了咱们魔改的时候了. 友情提醒,富文本编辑器 layedit 只有layui的V1版有,V2版没 ...

  10. Hexo博客美化之蝴蝶(butterfly)主题魔改

      Hexo是轻量级的极客博客,因为它简便,轻巧,扩展性强,搭建部署方便深受广大人们的喜爱.各种琳琅满路的Hexo主题也是被各种大佬开发出来,十分钦佩,向大佬仰望,大声称赞:流批!!! 我在翻看各种主 ...

随机推荐

  1. TOEFL | 202307 改革 · 新版题型总结

    目录 Listening(36min) Reading(35min) Speaking(16min) Writing(29min) Listening(36min) 2 conversation,3 ...

  2. python · pytorch | NN 训练常用代码存档

    1 pandas 读 csv import torch from torch import nn import numpy as np import pandas as pd from copy im ...

  3. 云服务器搭建自己的GitServer!

    云服务器搭建自己的GitServer! 如果你有一台云服务器并想在上面搭建自己的 Git 服务器,你可以使用 Git 自带的 git-shell ,也可以使用像 Gitea.GitLab.Gogs 这 ...

  4. [转帖]美国出口管制法律制度及中国企业风险防范——EAR核心内容解读

    http://bzy.scjg.jl.gov.cn/wto/zszc/myxgzs/202202/t20220221_636006.html 发布时间:2022-01-18 一.<美国出口管理条 ...

  5. [转帖]TiDB BR 备份至 MinIO S3 实战

    https://tidb.net/blog/3a31d41d#3.%E9%83%A8%E7%BD%B2%20MinIO%20S3%20%E5%8F%8A%E5%A4%87%E4%BB%BD%E6%81 ...

  6. [转帖]Kafka常见使用场景与Kafka高性能之道

    https://juejin.cn/post/6958997115012186119 消息队列使用场景 队列,在数据结构中是一种先进先出的结构,消息队列可以看成是一个盛放消息的容器,这些消息等待着各种 ...

  7. 【转帖】mysql一个索引块有多少指针_深刻理解MySQL系列之索引

    索引 查找一条数据的过程 先看下InnoDB的逻辑存储结构:node 表空间:能够看作是InnoDB存储引擎逻辑结构的最高层,全部的数据都存放在表空间中.默认有个共享表空间ibdata1.若是启用in ...

  8. [转帖]kubelet 原理解析五: exec的背后

    https://segmentfault.com/a/1190000022163850 概述 线上排查pod 问题一般有两种方式,kubectl log或者kubectl exec调试.如果你的 lo ...

  9. [转帖] jq实现json文本对比

      原创:打码日记(微信公众号ID:codelogs),欢迎分享,转载请保留出处. 简介# 近期,为了给一个核心系统减负,组内决定将一些调用量大的查询接口迁移到另一个系统,由于接口逻辑比较复杂,为了保 ...

  10. antv-x6 使用及总结

    1 简介 AntV是一个数据可视化(https://so.csdn.net/so/search?q=数据可视化&spm=1001.2101.3001.7020 )的工具(https://ant ...