题解 「THUPC 2022 初赛」造计算机
首先要最小化 ,其实这个应该打表
口胡了一些按顺序归位的做法均假了
然而排列的题先想着把二分图建出来
那么考虑分别对上面的每个环做归位
也就是将 从 换到 上去,这可以用一个寄存器做到
然而最后一个位置归位时会重复
那就撤销上一次操作,配合第二个寄存器完成一次 swap 即可
最后若两个寄存器反了还要再 swap 一下
点击查看代码
#include <bits/stdc++.h>
using namespace std;
#define INF 0x3f3f3f3f
#define N 100010
#define fir first
#define sec second
#define pb push_back
#define ll long long
//#define int long long
char buf[1<<21], *p1=buf, *p2=buf;
#define getchar() (p1==p2&&(p2=(p1=buf)+fread(buf, 1, 1<<21, stdin)), p1==p2?EOF:*p1++)
inline int read() {
int ans=0, f=1; char c=getchar();
while (!isdigit(c)) {if (c=='-') f=-f; c=getchar();}
while (isdigit(c)) {ans=(ans<<3)+(ans<<1)+(c^48); c=getchar();}
return ans*f;
}
int n;
int a[N];
bool vis[N];
int sta[N], top;
vector<pair<int, int>> ans;
void swap(int x, int y) {std::swap(a[x], a[y]); ans.pb({x, y});}
void dfs(int u) {
sta[++top]=u; vis[u]=1;
if (!vis[a[u]]) dfs(a[u]);
}
signed main()
{
n=read();
for (int i=1; i<=n; ++i) a[i]=read();
a[n+1]=n+1; a[n+2]=n+2;
for (int i=1; i<=n; ++i) if (a[i]!=i) goto jump;
puts("0 0"); return 0;
jump:
for (int i=1; i<=n; ++i) if (!vis[i] && a[i]!=i) {
top=0;
dfs(i);
if (top==1) continue;
for (int j=1; j<top; ++j) swap(sta[j], n+1);
swap(sta[top], n+2);
swap(sta[1], n+2);
swap(sta[top], n+1);
}
if (a[n+1]!=n+1) swap(a[n+1], a[n+2]);
// for (int i=1; i<=n+2; ++i) cout<<a[i]<<' '; cout<<endl;
printf("2 %d\n", ans.size());
for (auto it:ans) printf("%d %d\n", it.fir, it.sec);
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 无需6万激活码!GitHub神秘组织3小时极速复刻Manus,手把手教你使用OpenManus搭建本
· Manus爆火,是硬核还是营销?
· 终于写完轮子一部分:tcp代理 了,记录一下
· 别再用vector<bool>了!Google高级工程师:这可能是STL最大的设计失误
· 单元测试从入门到精通