继续来置换,这次的题也是很简单的。
题目大意:
对于一个长度为n的字符串,有一个数组表示第i个字符放到那个位置。
输入多个字符串,问这样操作k次后的字符串是什么样子的。
解题思路:
对于每一个置换,我们会求出循环节,这样可以减少模拟的次数。
下面是代码:
#include <stdio.h> #include <string.h> #include <iostream> #include <algorithm> #include <vector> #include <queue> #include <set> #include <map> #include <string> #include <math.h> #include <stdlib.h> #define clear(A, X, SIZE) memset(A, X, sizeof(A[0]) * (SIZE)) #define clearall(A, X) memset(A, X, sizeof(A)) #define max( x, y ) ( ((x) > (y)) ? (x) : (y) ) #define min( x, y ) ( ((x) < (y)) ? (x) : (y) ) using namespace std; struct node1 { int num,jie; } node[205]; char s[205]; bool vis[205]; int main() { int n,cnt,p,k; while(scanf("%d",&n),n) { for(int i=1; i<=n; i++) { scanf("%d",&node[i].num); } clearall(vis,false); for(int i=1; i<=n; i++) { if(!vis[i]); { p=node[i].num; cnt=1; vis[i]=true; while(p!=i) { cnt++; p=node[p].num; vis[p]=true; } node[i].jie=cnt; } } while(scanf("%d",&k),k) { gets(s); int len=strlen(s); for(int i=len; i<=n; i++) { s[i]=' '; } s[n+1]='\0'; clearall(vis,false); char c,t; for(int i=1; i<=n&&k; i++) { if(!vis[i]) { for(int j=0; j<k%node[i].jie; j++) { p=i; c=s[node[p].num]; s[node[p].num]=s[i]; p=node[p].num; vis[i]=true; while(p!=i) { vis[p]=true; t=s[node[p].num]; s[node[p].num]=c; c=t; p=node[p].num; } } } } puts(s+1); } puts(""); } return 0; }
原文地址:http://blog.csdn.net/lin375691011/article/details/38147983