题目描述:

小易在学校中学习了关于字符串的理论, 于是他基于此完成了一个字典的项目。

小易的这个字典很奇特, 字典内的每个单词都包含n个'a'和m个'z', 并且所有单词按照字典序排列。

小易现在希望你能帮他找出第k个单词是什么。

输入描述:

输入包括一行三个整数n, m, k(1 <= n, m <= 100, 1 <= k <= 109), 以空格分割。

输出描述:

输出第k个字典中的字符串,如果无解,输出-1。

输入样例:

2 2 6

输出样例:

zzaa

样例说明:

测试用例中字典里的字符串依次为aazz azaz azza zaaz zaza zzaa。

解题思路:

网易校招题。我一开始的想法就是暴力破解,先调用next_permutation来求全排列,然后将所有的字符串都存入unordered_map中,最后输出unordered_map中的第k个字符串即可,若无解则输出-1。然而我提交代码后出现了MLE 只通过30%的测试用例就出现内存超限啦。?

那没得一点办法 我只能把unordered_map去掉了,不再存储字符串,换成一个if语句,如果当前全排列的字符串是第k个就将它进行输出,如果无解就输出-1。然而这次提交代码后TLE了 通过50%的测试用例后就出现了运行超时。"100 100 1000000000"这个测试用例谁顶得住啊??

我哭了?今天运筹学考炸了,本来是想刷下题找成就感,结果这道题又是MLE又是TLE的。很显然网易不准C++玩家用next_permutation来暴力破解这道题。根据排列组合的知识可以知道,n个'a'和m个'z'只能组成 

小易的字典_测试用例

 个字符串,我在这里暂且把

小易的字典_测试用例

 表示为C(n,n+m)吧。先假设第一个字符为'a',那么剩下n-1个'a'和m个'z'组成的子序列能构成字典中的前C(n-1+m,n-1)个字符串。然后比较k和C(n-1+m,n-1),这里有俩种情况:①若k小于等于C(n-1+m,n-1),就说明第k个字符串是字典里前C(n-1+m,n-1)个中的一个(显然该字符串的第一个字符必为'a'),所以该问题又可以缩减为在子序列(n-1个'a'和m个'z')找到第k个字符串。②若k大于C(n-1+m,n-1),就说明结果字符串ans是以'z'开头的字符串中的第k-C(n-1+m,n-1)个字符串,所以该问题又可以缩减为在子序列(n个'a'和m-1个'z')找到第k-count(n+m-1,m-1)个字符串。while(n&&m)循环结束后,剩余子序列中只存在"aaa...a"或"zzz...z"的情况,若k不为1则说明无解输出-1,否则在字符串ans后补'a'或'z',最后输出ans即可。提交代码终于AC了!含泪AC?。

小易的字典_子序列_03

AC代码:MLE代码:

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

string init(int n,int m)    //初始化,得到字典中的第一个单词
{
    string ans = "";
    while(n--)
    {
        ans += 'a';
    }
    while(m--)
    {
        ans += 'z';
    }
    return ans;
}

int main()
{
    int n,m,k;   //字典中每个单词包含n个'a',m个'z',求第k个单词
    while(cin >> n >> m >> k)
    {
        string str = init(n,m);
        unordered_map<int,string> dictionary;   //小易的字典,key为单词,value为单词的次序
        int cnt = 0;    //计数器,记录当前字典的次序
        do{
            dictionary[++cnt] = str;   //存入字典中
        }while(next_permutation(str.begin(),str.end()));   //全排列
        cout << (k <= dictionary.size() ? dictionary[k] : "-1") << endl;   //输出字典中的第k个字符串,如果无解则输出-1
    }
    return 0;
}

AC代码:TLE代码:

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

string init(int n,int m)    //初始化,得到字典中的第一个单词
{
    string ans = "";
    while(n--)
    {
        ans += 'a';
    }
    while(m--)
    {
        ans += 'z';
    }
    return ans;
}

int main()
{
    int n,m,k;   //字典中每个单词包含n个'a',m个'z',求第k个单词
    while(cin >> n >> m >> k)
    {
        bool flag = false;    //判断有无解,默认为false无解
        string str = init(n,m);
        int cnt = 0;   //计数器,用来记录当前字典中的单词个数
        do{
            if(++cnt == k)
            {
                flag = true;  //有解
                cout << str << endl;    //输出字典中第k个字符串
                break;
            }
        }while(next_permutation(str.begin(), str.end()));    //全排列
        if(!flag)   //若无解,则输出-1
        {
            cout << -1 << endl;
        }
    }
    return 0;
}

AC代码:

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

int main()
{
    int n,m,k;   //字典中每个单词包含n个'a',m个'z',求第k个单词
    while(cin >> n >> m >> k)
    {
        string ans = "";
        while(n && m)
        {
            long long cnt = 1;   //一定要long long
            for(int i = 1; i < n; i++)    //求组合数
            {
                cnt *= (n+m-i);
                cnt /= i;
                if(cnt > k) break;   //防止越界,cnt>k就跳出for循环
            }
            if(k <= cnt)
            {
                ans += "a";
                n--;   //问题缩减为n-1个a和m个z中找第k个字符串
            }
            else
            {
                ans += "z";
                m--;
                k -= cnt;   //问题缩减为n-1个a和m个z中找第k-cnt个字符串
            }
        }
        if(k != 1)    //若k不为1说明无解
        {
            cout << -1 << endl;
        }
        else
        {
            //while(n&&m)循环结束后,剩余子序列中只存在"aaa...a"或"zzz...z"的情况
            while(n--)
            {
                ans += "a";
            }
            while(m--)
            {
                ans += "z";
            }
            cout << ans << endl;
        }
    }
    return 0;
}