Codeforces 936C

题意略。

思路:

这个题目没做出来是因为缺少一个整体的构造思路。

正确的构造思路是不断地在s中去构造并且扩大t的后缀,构造好的后缀总是放在前面,然后不断地把它往后挤,最后将s构造成t。

比如:

现在在s中构造好的t的后缀为a(在s中体现为前缀),包含在了s的前缀中,为了继续扩大这个t的后缀,我们需要将z添加到A前。

也就是AzB,要把z添加到A前,保持a不动,a为A的前缀,也是t的构造好的后缀。

AzB --- B`zA` --- AB`z --- zAB`    一共需要3次动作。

如果t的长度为len,那么我只需要3 * n次动作就可以完成了。

n最长又是2000,所以我们最多只需要6000次动作就可以了。小于6100,因此只要s和t的字符种类和个数一样就可以合法构造了。

 

详见代码:

#include<bits/stdc++.h>
using namespace std;

string s,t,s1,t1,str1,str2,str3;
int len;
vector<int> ans;

int main(){
    std::ios::sync_with_stdio(false);
    cin.tie(0);
    cout.tie(0);
    cin>>len;
    cin>>s>>t;
    s1 = s,t1 = t;
    sort(s1.begin(),s1.end());
    sort(t1.begin(),t1.end());
    if(s1 != t1){
        cout<<-1<<endl;
        return 0;
    }
    for(int i = len - 1;i >= 0;--i){
        str1 = str2 = str3 = "";
        bool jud = false;
        str2 = t[i];
        for(int j = 0;j < len;++j){
            if(jud){
                str3 += s[j];
                continue;
            }
            if(s[j] == t[i] && j >= (len - 1 - i)){
                jud = true;
                continue;
            }
            if(!jud) str1 += s[j];
        }
        reverse(str3.begin(),str3.end());
        s = str2 + str1 + str3;
        ans.push_back(len);
        ans.push_back(str1.length());
        ans.push_back(1);
    }
    cout<<3 * len<<endl;
    for(int i = 0;i < ans.size();++i){
        cout<<ans[i];
        if(i < ans.size() - 1) cout<<' ';
        else cout<<endl;
    }
    return 0;
}

 

posted @ 2018-08-07 20:56  温和的提比略  阅读(125)  评论(0编辑  收藏  举报