[PAT] A1067 Sort with Swap(0, i)

tech2022-09-02  141

题目大意

给出一个n个数的序列,数字为0~n-1的乱序,每次只能用数字0和另一个数交换。经过若干次能使序列变成有序的,问最少需要几次交换。

思路

贪心策略。 (1)每次让0占的位置上本来应该占的数字与0交换,交换结束后该数归位。 (2)但有可能数字0在自己的位置上(即第0位),但仍有其他数字没有归位。则此时将0与其交换,让它暂时占据0位,重复步骤(1)。 优化: 在(2)中,如果每次都循环去找没有归位的数字,若那个数字在很后面,则要判断很多次,测试点1、2会超时。因此最外层循环是i从数字0到数字n-1,每一次循环结束后,数字i的状态可能有两种,一种是在0不断“让位”中归位(即它与0处在同一个错位圈中);另一种是0不断“让位”也不能导致它归位,则强制让它暂时占据0位,在下一次循环的时候,数字0要想回到0位,必须通过与它交换才可以(则在代码中的while循环内实现),而0与它交换的条件只能是因为0占据了它本来的位置,则它必然能在下一次循环中归位。

AC代码

#define _CRT_SECURE_NO_WARNINGS #include<iostream> using namespace std; #define MAX 100000 int main() { int i, temp, times = 0, n, locate[MAX]; scanf("%d", &n); for (i = 0; i < n; i++) { scanf("%d", &temp); locate[temp] = i; } for (i = 1; i < n; i++) { while (locate[0] != 0) { temp = locate[locate[0]]; locate[locate[0]] = locate[0]; locate[0] = temp; times++; } //如果每次判断0归位后,再寻找没归位的数字,就会多余循环判断,引起超时。 if (locate[i] != i) {//此时代表数字0已归位,但是数字i仍然不在自己的位置上 //则需要交换0和i,也即数字0的位置和数字i的位置对调 temp = locate[i]; locate[i] = locate[0]; locate[0] = temp; times++; } } printf("%d", times); return 0; }
最新回复(0)