Live2D

Solution -「ARC 110F」Esoswap

Description

  Link.

  给定 0n1 的排列 p0..n1,每次操作给出 i,交换 pip(i+pi)modn。构造一种使排列升序的操作序列。

  n100

Solution

  反正兔子就一个样例观察法,一个暴力伪解拍上去就 AC 了。(

  先讲讲我的伪解,观察样例解释:

  • First, announce i=6. We swap P6(=5) and P(6+5)mod8=P3(=6), turning P into 7,1,2,5,4,0,6,3.

  • Then, announce i=0. We swap P0(=7) and P(0+7)mod8=P7(=3), turning P into 3,1,2,5,4,0,6,7.

  • Then, announce i=3. We swap P3(=5) and P(3+5)mod8=P0(=3), turning P into 5,1,2,3,4,0,6,7.

  • Finally, announce i=0. We swap P0(=5) and P(0+5)mod8=P5(=0), turning P into 0,1,2,3,4,5,6,7.

  

  发现三次 pi=5,一次 pi=7。考虑到样例的迷惑性,我们尝试让对 pi=5 的操作挨在一起。交换第一步和第二步,发现操作序列仍合法。

  接下来,我们强行解释该操作序列的内在逻辑:

  • 希望 p7=7,反复操作 pi=7 直到 p7=7
  • 希望 p7=7p6=6,反复操作 pi=6(样例中不需要操作),直到不满足 p7=7 回到第一步,或满足 p6=6
  • 希望 p5=5p6=6p7=7,操作同上。
  • ……

  综上,交换策略为

选择不满足 pi=i 的最大的 pi 进行交换直到序列升序。

  然后就 AC 了,复杂度不知道。(


  正解是先操作使得 p={n1,n2,,0} 然后逆序。操作方法考察从后往前的每个 i,不断操作 i 直至 pi=ni1,可证至多操作 O(n) 次,总复杂度 O(n2)

Code

  伪解:

/* Clearink */
 
#include <cstdio>
#include <vector>
 
#define rep( i, l, r ) for ( int i = l, rpbound##i = r; i <= rpbound##i; ++i )
#define per( i, r, l ) for ( int i = r, rpbound##i = l; i >= rpbound##i; --i )
 
const int MAXN = 100;
int n, p[MAXN + 5];
std::vector<int> ans;
 
inline void iswp ( int& a, int& b ) { a ^= b ^= a ^= b; }
 
int main () {
	scanf ( "%d", &n );
	rep ( i, 0, n - 1 ) scanf ( "%d", &p[i] );
	while ( true ) {
		int irr = -1;
		rep ( i, 0, n - 1 ) if ( i ^ p[i] && ( !~irr || p[i] > p[irr] ) ) irr = i;
		if ( !~irr ) break;
		ans.push_back ( irr );
		iswp ( p[irr], p[( irr + p[irr] ) % n] );
	}
	printf ( "%d\n", ( int ) ans.size () );
	for ( int i: ans ) printf ( "%d\n", i );
	return 0;
}

  正解:

/* Clearink */

#include <cstdio>
#include <vector>

#define rep( i, l, r ) for ( int i = l, rpbound##i = r; i <= rpbound##i; ++i )
#define per( i, r, l ) for ( int i = r, rpbound##i = l; i >= rpbound##i; --i )

const int MAXN = 100;
int n, p[MAXN + 5];
std::vector<int> ans;

inline void iswp ( int& a, int& b ) { a ^= b ^= a ^= b; }
inline void oper ( const int i ) {
	ans.push_back ( i ), iswp ( p[i], p[( i + p[i] ) % n] );
}

int main () {
	scanf ( "%d", &n );
	rep ( i, 0, n - 1 ) scanf ( "%d", &p[i] );
	per ( i, n - 1, 1 ) for ( ; p[i] != n - i - 1; oper ( i ) );
	per ( i, n - 2, 0 ) {
		rep ( j, i + 1, n - 2 ) oper ( j );
		oper ( i ), oper ( i );
	}
	printf ( "%d\n", ( int ) ans.size () );
	for ( int i: ans ) printf ( "%d\n", i );
	return 0;
}

posted @   Rainybunny  阅读(55)  评论(0编辑  收藏  举报
编辑推荐:
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
阅读排行:
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现
点击右上角即可分享
微信分享提示