POJ1159解题心得

题目:http://poj.org/problem?id=1159

刚开始,从样例的特征去思考。总让我从回文数的角度去思考,想出几个方案,可都用了数据去检验,发现不行。如:ABCDDCB,BACDCABD等样例。

然后,思考半天,没想出可行的方案。于是,看人家的解题报告,发现如下公式:

此题所需的解 = 原字符串的长度 — 原字符串和逆字符串的最长公共子序列的长度

于是,啪啦啪啦把代码写完,测试,没问题。提交,发现,Rumtime Error。发现错误,动态规划的数组开辟空间过大,放在函数里面,把它放到函数外就OK了。然后,再次提交,发现,Memory Limit Exceed。解决办法,把动态规划的dp数组压缩空间,变成滚动数组。于是,AC代码就变成如下的样子了。

#include<iostream>
#include<stdio.h>
#include<string.h>
#include<stdlib.h>
using namespace std;
#define INF 88888888

#define MAX_N 5000
int length;
char originalStr[MAX_N + 1];
int dp[2][MAX_N + 1];//dp[i % 2][j]:=s1...si和s1...sj的LCS的长度

void inputTestData(){
    cin >> length;
    for(int i=0;i<length;i++){
        cin >> originalStr[i];
    }
}

char* reverseStr(char* str,int length){
    char* p = (char*)malloc(sizeof(char) * length);
    for(int i=0;i<length;i++){
        *(p + i) = *(str + length - 1 - i);
    }
    return p;
}


int calcLongestCommonSubsequenceLength(char* str1,char* str2){
    for(int i=0;i<length;i++){
        for(int j=0;j<length;j++){
            if(str1[i] == str2[j]){
                dp[(i + 1) % 2][j + 1] = dp[i % 2][j] + 1;
            }
            else{
                dp[(i + 1) % 2][j + 1] = max(dp[(i + 1) % 2][j],dp[i % 2][j + 1]);
            }
        }
    }
    return dp[length % 2][length];
}

void solve(){
    inputTestData();
    char* reversedStr = reverseStr(originalStr,length);
    int lengthOfLcs = calcLongestCommonSubsequenceLength(originalStr,reversedStr);
    int result = length - lengthOfLcs;
    cout << result << endl;
}

int main(void){
    solve();
    return 0;
}

附:使用C/C++出现Runtime Error的情况,基本都是如下原因:

1.数组越界;(本质上还是第二个原因)

2.访问没有权限访问的内容,即没有为指针分配足够的空间;

3.指针的值是NULL,即常说的“空指针”。

如果语言是Java,出现任何Exception或Error都会得到Runtime Error的结果。

常见的Exception或Error:

空指针异常,数组越界异常,类型转换异常,除0异常。

栈溢出错误(通常由无穷递归引起,也可能是方法内开辟的数组空间过大),断言错误(由于OJ无法看到错误栈的信息,所以,在算法竞赛中,不要用assert这个关键字)。

转载于:https://www.cnblogs.com/fuzhihong0917/p/6389056.html

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值