解题思路:

     1.impossible的情况:如果有一个字符出现的次数是奇数次数,而且n是偶数,那么不可能构成回文
       如果n是奇数,但是已经有一个字符出现的次数是奇数次数了,那么如果又有一个字符是奇数次数,
 就不可能构成回文。

     2.如果n是奇数,计算中间那个字符交换的次数的时候,不需要模拟把这个数移动到中间去,因为移动到中 间的话假设有一对数都在左边或者都         在右边,那么交换成回文的时候就要经过中间,就会每次把cnt多加了1,而这个1是没有必要的,因为可以所有的回文移动完了之后再把这个           独立的奇数移动过去,才能保证交换次数最少。

    3.参见大牛的博客https://blog.csdn.net/liuchuo/article/details/51990430

注意事项:

    注意分类n为奇数和偶数吧   

参考代码:

#include <iostream>
#include <cstdio>
using namespace std;
int main() {
   int n;
   scanf("%d",&n);
   char a[10000];
   scanf("%s",a);
   int j=n-1;//j指向字符串需要交换的末尾下标
   int count=0;//记录交换次数
   int flag=0;//记录是否有字符只出现了单次,
                // 未找到相同字符串时为1
    for (int i = 0; i < j ; i++) {//i指针从头遍历到倒数第二个字符
        for (int k = j; k >=i ; k--) {//k指针从后面往前一直到i寻找和s[i]相同的s[k]
            if(k==i){//如果找不到相同的
                if(n%2==0||flag==1){//impossible的两种情况
                    printf("Impossible");
                    return 0;
                }
                flag=1;
                count+=(n-1)/2-i;  //移动到中间需要的交换次数(距离即为交换次数),
                                    // (不需要现在交换过去,最后交换)
                                    // 此时n一定为奇数,因为n是偶数已经直接输出Impossible了
            }
            else{
                if(a[k]==a[i]){
                    for(int l=k;l<j;l++){
                        swap(a[l],a[l+1]);//把s[k]换到s[j]处
                        count++;//统计交换次数
                    }
                    j--;
                    break;
                }
            }

        }
    }
    printf("%d",count);
    return 0;
}


点赞(1)
 

0.0分

2 人评分

C语言网提供由在职研发工程师或ACM蓝桥杯竞赛优秀选手录制的视频教程,并配有习题和答疑,点击了解:

一点编程也不会写的:零基础C语言学练课程

解决困扰你多年的C语言疑难杂症特性的C语言进阶课程

从零到写出一个爬虫的Python编程课程

只会语法写不出代码?手把手带你写100个编程真题的编程百练课程

信息学奥赛或C++选手的 必学C++课程

蓝桥杯ACM、信息学奥赛的必学课程:算法竞赛课入门课程

手把手讲解近五年真题的蓝桥杯辅导课程

评论列表 共有 0 条评论

暂无评论