快速幂思想
题目描述
牛牛拿到了一个长度为N的排列和M个区间,一开始排列是1、2、3......N。
然后他将这些区间在按顺序在排列上翻转,全部翻转一遍称一次操作。
现在他要去搞文化了...所以拜托你告诉他经过K次操作后的排列长什么样子。
题目分析
刚刚开始看题目的时候一脸懵逼,没看懂题目的意思,不过想想毕竟是第一题应该简单一些,所以就再次再次的看题目,发现:m次操作代表的是一次总的操作,也就是说要做k次m次的操作就是最终答案,但是题目给出的范围很大,所以需要用倍增的思想,倍增的思想能很快地实现计算,所以我们引进快速幂思想去分析题目
#include <bits/stdc++.h> using namespace std; const int maxn = 1e6 + 10; int ans[maxn], temp[maxn], mode[maxn]; int n, m, k; void solve(int a[], int b[]) { for (int i = 1; i <= n; ++i) temp[i] = b[i]; //防止a和b为同一数组时修改自身。 for (int i = 1; i <= n; ++i) a[i] = temp[a[i]]; } void fastpow() { int t = k; while (t) { if (t & 1) solve(ans, mode); solve(mode, mode); t >>= 1; } } int main() { scanf("%d%d%d", &n, &m, &k); for (int i = 1; i <= n; ++i) ans[i] = mode[i] = i; int l, r; while (m--) { scanf("%d%d", &l, &r); while (l < r) swap(mode[l++], mode[r--]); } fastpow(); for (int i = 1; i <= n; ++i) printf("%d ", ans[i]); return 0; }