USACO Calf Flac题解

阅读更多

这一题有点烦

我一开始的思路是,回文序列么,就是正序字符串和逆序字符串中相同的那一串

于是乎,就转化成求最长公共子字符串,于是用动态规划,O(N^2)的时间复杂度和空间复杂度

首先是内存超了,于是换成O(n)空间复杂度的实现方式,即只记录上一状态就可以

接着到最后一个测试程序的时候,时间也超了

 

无奈,想不出其他思路的情况下,看了NOCOW的解题,O(n)的动态规划

思路是这样的:

写道
begin
if cha[i]=cha[i-1] then f[i]:=i-1;
if cha[i]=cha[f[i-1]-1] then f[i]:=f[i-1]-1;
end;

 

以下是代码,注释部分是我一开始的思路:

/*
ID: bbsunch2
PROG: calfflac
LANG: C++
*/
#include 
#include 
#include 
#include 
#include 
#include 

using namespace std;

int main()
{
    ofstream fout ("calfflac.out");
    ifstream fin ("calfflac.in");
    string input = "";
    while(fin.good() && !fin.eof())
    {
        string line;
        getline(fin,line,'\n');
        input = input + line + "\n";
    }
    //cout << input << endl;
    //建立字符串中的字母和原始字符串位置对应关系
    int charIndicator = -1;
    vector correspondingPosition;
    string letterString = "";
    for(int i = 0; i < input.size(); i++)
    {
        char single = input[i];
        if(single >= 'A' && single <= 'Z')
        {
            charIndicator++;
            letterString += single;
            correspondingPosition.push_back(charIndicator);
        }else if(single >= 'a' && single <='z')
        {
            single = single - 32;
            charIndicator++;
            letterString += single;
            correspondingPosition.push_back(charIndicator);
        }else
        {
            charIndicator++;
        }
    }
    vector f;
    f.push_back(0);
    for(int i = 1; i < letterString.size(); i++)
    {
        if(letterString[i] == letterString[f[i-1]])
        {
            f.push_back(i-1);
            if(letterString[i] == letterString[f[i-1]-1])
            {
                f[i] = (f[i-1]-1);
            }
        }else if(f[i-1] - 1 >=0)
        {
            if(letterString[i] == letterString[f[i-1]-1])
            {
                f.push_back(f[i-1]-1);
            }else
            {
                f.push_back(i);
            }
        }else
        {
            f.push_back(i);
        }
    }
    int maxI = 0;
    int maxStep = 0;
    for(int i = 1; i < letterString.size(); i++)
    {
        if(maxStep < i-f[i]+1)
        {
            maxStep = i - f[i]+1;
            maxI = i;
        }
    }
    /*string reverseLetterString = "";
    for(int i = letterString.size()-1; i >= 0; i--)
    {
        reverseLetterString += letterString[i];
    }
    //vector > matrix;
    int maxI = 0;
    int maxStep = 0;
    vector line1;
    vector line2;
    for(int i = 0; i < letterString.size(); i++)
    {
        line1.push_back(0);
        line2.push_back(0);
    }



    for(int i = 0; i < letterString.size(); i++)
    {
        vector lineInMatrix;
        for(int k = 0; k < letterString.size(); k++)
        {
            lineInMatrix.push_back(0);
        }
        matrix.push_back(lineInMatrix);
    }


    for(int k = 0; k < reverseLetterString.size(); k++)
    {
        for(int i = 0; i < letterString.size(); i++)
        {
            line2[i] = 0;
            if(letterString[i] == reverseLetterString[k])
            {
                if(i == 0 || k == 0)
                {
                    line2[i] = 1;
                }else
                {
                    line2[i] = line1[i-1] + 1;
                }
                if(maxStep <= line2[i])
                {
                    maxStep = line2[i];
                    maxI = i;
                }
            }
        }
        for(int index = 0; index < letterString.size(); index++)
        {
            line1[index] = line2[index];
        }

    }*/
    fout << maxStep << endl;
    int startLetterPosition = maxI - maxStep + 1;
    int startPosition = correspondingPosition[startLetterPosition];
    int endPosition = correspondingPosition[maxI];

    string sub = input.substr(startPosition, endPosition - startPosition + 1);

    fout << sub << endl;

    return 0;
}

 

你可能感兴趣的:(USACO,算法)