CF1117E Decypher the String

交互题。

这题需要一个很精妙的构造,所以写一篇题解来加深自己的理解。

Solution

如果直接看见 \(3\) 次机会和 \(10000\)\(|s|\) 可能会很迷茫,但是还有一个 \(26\) ,那我们就可以写一个 \(26^2<10000<26^3\) ,然后就到了精妙的构造了。

第一次机会:

\(aaa...aaa(26*26)bbb...bbb(26*26)ccc...ccc(26*26)...\)

每一个 \(26*26\) 是大块

第二次机会:

\(aaa...aaa(26)bbb...bbb(26)ccc...ccc(26)...\)

每一个 \(26\) 是小块

第三次机会:

\(abcdefg...xyzabcdefg...xyz...\)

这样对于变化后的串 \(s’\) 的每一位都可以找到大块中的小块中的准确位置,即整个原串。

(或者这样理解:将 \(s'\) 拆开,三次操作相当于找到位置 \(i\) 所对应的26进制数)

代码

#include<bits/stdc++.h>

using namespace std;
const int N=10005;
int pos[N];
string s,print,res;
char ans[N];

int main(){
    cin>>res;
    int len=res.size();
    print="? ";
    char ch='a';
    //先分成26*26的大块,确定在大块的位置
    for(int i=0;i<len;i++){
        print+=ch;
        if((i+1)%(26*26)==0) ch++;
    }
    cout<<print<<endl;
    cin>>s;
    for(int i=0;i<len;i++) pos[i]+=(s[i]-'a')*26*26;
    ch='a';
    print="? ";
    //分成26的小块,确定在小块的位置
    for(int i=0;i<len;i++){
        print+=ch;
        if((i+1)%26==0) ch++;
        if(ch=='z'+1) ch='a';
    }
    cout<<print<<endl;
    cin>>s;
    for(int i=0;i<len;i++) pos[i]+=(s[i]-'a')*26;
    ch='a';
    print="? ";
    //分成1,确定最终的位置
    for(int i=0;i<len;i++){
        print+=ch;
        ch++;
        if(ch=='z'+1) ch='a'; 
    }
    cout<<print<<endl;
    cin>>s;
    for(int i=0;i<len;i++) pos[i]+=(s[i]-'a');
    for(int i=0;i<len;i++)
        ans[pos[i]]=res[i];
    printf("! %s",ans);
    fflush(stdout);
    return 0;
}
posted @ 2020-09-21 11:13  jasony_sam  阅读(152)  评论(0编辑  收藏  举报