小兔的话

欢迎大家在评论区留言哦~


HH去散步

题目限制

  • 内存限制:125.00MB
  • 时间限制:1.00s
  • 标准输入
  • 标准输出

题目知识点

  • 动态规划 \(dp\)
  • 矩阵
    • 矩阵乘法
    • 矩阵加速
    • 矩阵快速幂
  • 思维
    • 构造

题目来源

「 洛谷 」P2151 [SDOI2009]HH去散步


为了方便大家阅读通畅,题目可能略有改动,保证不会造成影响

题目

题目背景

HH 有个一成不变的习惯,喜欢在饭后散步,就是在一定的时间内,走一定的距离

同时, HH 是一个喜欢变化的人,她不会立刻沿着刚刚走过来的路走回去,她也希望每天走过的路径都不完全一样,她想知道每一天他究竟有多少种散步的方法

题目描述

现在 HH 送给你一张学校的地图,请你帮助她求出从地点 \(A\) 走到地点 \(B\) 一共有多少条长度为 \(T\) 的散步路径(答案对 \(45989\) 取模)

格式

输入格式

输入共 \(M + 1\) 行:

第 \(1\) 行:输入 \(5\) 个整数 \(N, \ M, \ T, \ A, \ B\);\(N\) 表示 学校里的路口的个数(编号为 \(0 \sim N - 1\)),\(M\) 表示 学校里的道路的条数,\(T\) 表示 HH 想要散步的距离,\(A\) 表示 散步的出发点, \(B\) 表示 散步的终点

接下来 \(M\) 行:每行 \(2\) 个用空格隔开的整数 \(u_i, \ v_i\);表示 长度为 \(1\) 的第 \(i\) 条路 连接 路口 \(u_i\)路口 \(v_i\)

输出格式

输出共一行:表示你所求出的答案(对 \(45989\) 取模)

样例

样例输入

4 5 3 0 0
0 1
0 2
0 3
2 1
3 2

样例输出

4

提示

数据范围

对于 \(30 \%\) 的数据:满足 \(N \leq 4, \ M \leq 10, \ T \leq 10\)

对于 \(100 \%\) 的数据:满足 \(N \leq 50, \ M \leq 60, \ T \leq 2 ^ {30}, \ u_i \neq v_i\)


思路

这道题如果没有 她不会立刻沿着刚刚走过来的路走回去 的限制,就可以根据点与点的关系先构造出一个 \(n * n\) 的矩阵 \(\mathrm{x}\)(\(\mathrm{x}[i][j]\) 表示从 \(i\) 走 \(1\) 步到 \(j\) 的方案数),累乘 \(T\) 次(就是走了 \(T\) 步),就用矩阵快速幂优化既可以通过了

现在就考虑加上这句话的限制后如何构造矩阵了


分析

考虑矩阵定义大致不变,即 \(\mathrm{x}[i][j]\) 表示从 \(i\) 走 \(1\) 步到 \(j\) 的方案数

由于有限制,就要记录刚刚走过来的路是哪一条

不妨把每条边对应的 \(u_i\) 和 \(v_i\) 拆成两个二元组 \(\mathrm{(node, id)}\),表示刚刚从第 \(\mathrm{id}\) 条路走到 \(\mathrm{node}\),也就是每条无向边 \((u_i \leftrightarrow, v_i)\) 分成两条有向边 \((u_i \to v_i)\) 和 \((v_i \to u_i)\),其中 \(\mathrm{node}\) 表示当前这条有向边的终点,\(\mathrm{id}\) 表示与之对应的无向边的编号

那么 \(\mathrm{x}[i][j] = 1\) 定义就是 第 \(i\) 个二元组 走 \(1\) 步到 第 \(j\) 个二元组 的方案数

其值只可能为 \(0\) 或 \(1\)(因为只走了 \(1\) 步),其中值为 \(1\) 的条件就是 \(\mathrm{id}_i \neq \mathrm{id}_j\) 且 \(\mathrm{node}_i\) 与 \(\mathrm{node}_j\) 有一条边

推出了矩阵,但是还有一个细节,就是第一步的方案数

起始点是没有上一条边的,所以需要预处理一下(这里相当于先走了一次)

预处理矩阵 \(\times\) 矩阵快速幂(\(T - 1\) 次,预处理走了一次)就可以得到最终的矩阵了

最后把 起始点(超级源点)终点(可能有多个,因为分了边) 的路径加起来取模就可以了


代码

#include <cstdio>
#include <cstring> int rint()
{
int x = 0, fx = 1; char c = getchar();
while (c < '0' || c > '9') { fx ^= ((c == '-') ? 1 : 0); c = getchar(); }
while ('0' <= c && c <= '9') { x = (x << 3) + (x << 1) + (c ^ 48); c = getchar(); }
if (!fx) return -x;
return x;
} const int MOD = 45989; const int MAX_N = 20;
const int MAX_M = 60; int N, M, T, A, B, node;
int e[MAX_M * 2 + 5][3]; struct Matrix
{
int mx[MAX_M * 2 + 5][MAX_M * 2 + 5]; Matrix () { memset(mx, 0, sizeof(mx)); } void init() { for (int i = 0; i <= node; i++) mx[i][i] = 1; } Matrix operator * (const Matrix &rhs) const
{
Matrix res;
for (int i = 0; i <= node; i++)
for (int j = 0; j <= node; j++)
for (int k = 0; k <= node; k++)
res.mx[i][j] = (res.mx[i][j] + mx[i][k] * rhs.mx[k][j]) % MOD;
return res;
}
} dp, quick; Matrix qpow(Matrix mx, int k)
{
Matrix res; res.init();
while (k > 0)
{
if (k & 1) res = res * mx;
mx = mx * mx; k >>= 1;
}
return res;
} int main()
{
N = rint(), M = rint(), T = rint();
A = rint() + 1, B = rint() + 1;
for (int i = 1; i <= M; i++)
{
e[i][0] = rint() + 1, e[i][1] = rint() + 1;
e[i + M][0] = e[i][1], e[i + M][1] = e[i][0];
if (e[i][0] == A) ++dp.mx[0][i];
if (e[i + M][0] == A) ++dp.mx[0][i + M];
}
node = M << 1;
for (int i = 1; i <= node; i++)
for (int j = 1; j <= node; j++)
if (i + M != j && i - M != j && e[i][1] == e[j][0]) ++quick.mx[i][j];
int ans = 0;
Matrix res = dp * qpow(quick, T - 1);
for (int i = 1; i <= node; i++)
if (e[i][1] == B) ans = (ans + res.mx[0][i]) % MOD;
printf("%d\n", ans);
return 0;
}

「 洛谷 」P2151 [SDOI2009]HH去散步的更多相关文章

  1. 洛谷P2151 [SDOI2009] HH去散步 [矩阵加速]

    题目传送门 HH去散步 题目描述 HH有个一成不变的习惯,喜欢饭后百步走.所谓百步走,就是散步,就是在一定的时间 内,走过一定的距离. 但是同时HH又是个喜欢变化的人,所以他不会立刻沿着刚刚走来的路走 ...

  2. [bzoj1875] [洛谷P2151] [SDOI2009] HH去散步

    Description HH有个一成不变的习惯,喜欢饭后百步走.所谓百步走,就是散步,就是在一定的时间 内,走过一定的距离. 但 是同时HH又是个喜欢变化的人,所以他不会立刻沿着刚刚走来的路走回. 又 ...

  3. 洛谷 P2151 [SDOI2009]HH去散步

    题目链接 思路 如果没有不能走上一条边的限制,很显然就是dp. 设f[i][j]表示到达i点走了j步的方案数,移到k点可以表示为f[k][j+1]+=f[i][j]. 如果有限制的话,可以考虑用边表示 ...

  4. P2151 [SDOI2009]HH去散步

    题目描述 HH有个一成不变的习惯,喜欢饭后百步走.所谓百步走,就是散步,就是在一定的时间 内,走过一定的距离. 但是同时HH又是个喜欢变化的人,所以他不会立刻沿着刚刚走来的路走回. 又因为HH是个喜欢 ...

  5. Luogu P2151 [SDOI2009]HH去散步 矩乘加速DP

    思路:矩乘优化DP 提交:3次(用了一个奇怪的东西导致常数过大) 题解: 如果可以走完正向边后又走反向边那就显然了,但是不能走,所以我们要将正反向边分别编号,区分正反向边. 所以这道题的矩阵是以边的编 ...

  6. AC日记——[SDOI2009]HH去散步 洛谷 P2151

    [SDOI2009]HH去散步 思路: 矩阵快速幂递推(类似弗洛伊德): 给大佬跪烂-- 代码: #include <bits/stdc++.h> using namespace std; ...

  7. 「 洛谷 」P2768 珍珠项链

    珍珠项链 题目限制 内存限制:125.00MB 时间限制:1.00s 标准输入输出 题目知识点 动态规划 \(dp\) 矩阵 矩阵乘法 矩阵加速 矩阵快速幂 题目来源 「 洛谷 」P2768 珍珠项链 ...

  8. 「 洛谷 」P4539 [SCOI2006]zh_tree

    小兔的话 推荐 小兔的CSDN [SCOI2006]zh_tree 题目限制 内存限制:250.00MB 时间限制:1.00s 标准输入输出 题目知识点 思维 动态规划 \(dp\) 区间\(dp\) ...

  9. bzoj1875: [SDOI2009]HH去散步

    终于A了...早上按自己以前的写法一直WA.下午换了一种写法就A了qwq #include<cstdio> #include<cstring> #include<iost ...

随机推荐

  1. show engine innodb status 输出结果解读

    show engine innodb status 输出结果解读 基于MySQL 5.7.32 最近想整理一下show engine innodb status的解读,但是发现中文互联网上相关的信息要 ...

  2. Webpack的理解以及解决了的问题

    一.背景 Webpack 最初的目标是实现前端项目的模块化,旨在更高效地管理和维护项目中的每一个资源 模块化 最早的时候,我们会通过文件划分的形式实现模块化,也就是将每个功能及其相关状态数据各自单独放 ...

  3. Standalone模式下,通过Systemd管理Flink1.11.1的启停及异常退出

    Flink以Standalone模式运行时,可能会发生jobmanager(以下简称jm)或taskmanager(以下简称tm)异常退出的情况,我们可以使用Linux自带的Systemd方式管理jm ...

  4. UVA11292杀怪

    题意:      一个怪物有N个头,每个头都有半径,然后有M个骑士,每个骑士能砍掉半径小于等于 X[i]的头,花费为X[i],并且一个骑士只能用一次,问砍掉怪物所有头的最小花费. 思路:       ...

  5. hdu4990 矩阵快速幂

    题意:       给你一短代码,让你优化这个代码,代码如下 #pragma comment(linker, "/STACK:1024000000,1024000000") #in ...

  6. CTFHub-技能树-信息泄露

    CTFHub-技能树-信息泄露 1.目录遍历 文件夹不多,直接手翻就行 2.PHPinfo 直接搜索ctfhub{就能找到 3.备份文件下载 3-1 网站源码 打开之后: 因为这里组合比较少可以手动测 ...

  7. 还不懂 redis 持久化?看看这个

    Redis 是一个内存数据库,为了保证数据不丢失,必须把数据保存到磁盘,这就叫做持久化. Redis 有两种持久化方法: RDB 方式以及 AOF 方式 RDB 持久化 前言 RDB持久化把内存中的数 ...

  8. Jetpack Compose What and Why, 6个问题

    Jetpack Compose What and Why, 6个问题 1.这个技术出现的背景, 初衷, 要达到什么样的目标或是要解决什么样的问题. Jetpack Compose是什么? 它是一个声明 ...

  9. Envoy :V3APi 开启 TLS

    方案架构 本次实例与官方Envoy front_proxy Example相似,首先会有一个Envoy单独运行.ingress的工作是给其他地方提供一个入口.来自外部的传入连接请求到这里,前端代理将会 ...

  10. [其他] vscode 快速教程

    概述 vs:集成开发环境,包括软件生命周期中需要的大部分工具,如UML,代码管控,IDE等 vs code:代码编辑器,支持插件扩展,对网页和云端开发做了优化 快捷键 F1/ctrl+shift+p: ...