问题:
给定一个数字序列A1,A2,…,An,求i,j(1<=i<=j<=n),使得Ai+····+Aj最大,输出这个最大和。
步骤:
#include
using namespace std;
const int maxn = 10010;
int A[maxn],dp[maxn];//A[i]存放序列,dp[i]存放以A[i]结尾的连续序列的最大和
int main(){
int n;
scanf("%d",&n);
for(int i =0;i<n;i++){
scanf("%d",&A[i]);//读入序列
}
//边界
dp[0] = A[0];
for(int i =1;i<n;i++){
//状态转移方程
dp[i] = max(dp[i-1]+A[i],A[i]);
}
//遍历得到最大的dp[i]
int k = 0;
for(int i =0;i<n;i++){
if(dp[i]>dp[k]){
k = i;
}
}
printf("%d\n",dp[k]);
return 0;
}
问题:
在一个数字序列中,找到一个最长的子序列(可以不连续),使得这个子序列是不下降(非递减)的。
步骤:
令dp[i]表示以A[i]结尾的最长不下降子序列长度。这样对A[i]来说有两种可能:
只要让i从小到大遍历即可求出整个dp数组,之后求出整个dp数组中的最大值即是整个序列的LIS长度。
#include
using namespace std;
const int N = 100;
int A[N],dp[N];
int main(){
int n ;
scanf("%d",&n);
for(int i =0;i<n;i++){
scanf("%d",&A[i]);
}
int ans = -1;//记录最大的dp[i]
for(int i =0;i<n;i++){
dp[i] = 1;//初始化,只包含自己的LIS
for(int j = 0;j<n;j++){
if(A[j]< = A[i] && (dp[j]+1>dp[i])){
dp[i] = dp[j]+1;//状态转移方程
}
}
if(dp[i] > ans){
ans = dp[i];
}
}
printf("%d\n",ans);
return 0;
}
问题:
给定两个字符串(或数字序列)A和B,求一个字符串,使得这个字符串是A和B的最长公共部分(子序列可以不连续)。
令dp[i][j]表示A中i号位和B中j号位之前的LCS长度,那么可以根据A[i]和B[j]的情况,分为两种决策:
#include
using namespace std;
const int N = 100;
char A[N],B[N];
int dp[N][N];
int main(){
int n;
gets(A+1);//从下标为1开始读入
gets(B+1);
int lenA = strlen(A+1);//读取长度也从+1开始
int lenB = strlen(B+1);
//边界
for(int i =0;i<lenA;i++){
dp[i][0] = 0;
}
for(int j =0;j<lenB;j++){
dp[0][j] = 0;
}
//状态转移方程
for(int i =0;i<lenA;i++){
for(int j =0;j<lenB;j++){
if(A[i]==B[j]){
dp[i][j] = dp[i-1][j-1]+1;
}else{
dp[i][j] = max(dp[i-1][j],dp[i][j-1]);
}
}
}
//dp[lenA][lenB]是答案
printf("%d\n",dp[lenA][lenB]);
return 0;
}
问题:
给出一个字符串S,求S的最长回文子串的长度。
举例:
字符串“PATZJUJZTACCBCC”的最长回文子串为“ATZJUJZTA”,长度为9
步骤:
令dp[i][j]表示S[i]至S[j]所表示的子串是否是回文子串,是则为1,不是则为0。这样根据S[i]是否等于S[j],可以把转移情况分为两类:
由于如果按照i和j从小到大的顺序来枚举子串的两个端点,然后更新dp[i][j],会无法保证dp[i+1][j-1]已经被计算过,从而无法得到正确的dp[i][j]。
注意到边界是长度为1和2的子串,且每次转移时都对子串的长度-1,因此不妨考虑到子串的长度和初始位置进行枚举。
#include
using namespace std;
const int maxn = 1010;
char S[maxn];
int dp[maxn][maxn];
int main(){
gets(S);
int len = strlen(S),ans = -1;
memset(dp,0,sizeof(dp));//初始化dp为0
//边界
for(int i =0;i<len;i++){
dp[i][i] = 1;
if(i < len-1){
if(S[i] == S[i+1]){
dp[i][i+1] = 1;
ans = 2;//初始化时注意当前最长回文子串长度
}
}
}
//状态转移方程
for(int L = 3;L<=len;L++){//枚举子串的长度
for(int i =0;i+L-1<len;i++){//枚举子串的起始端点
int j = i+L-1;//子串的右端点
if(S[i] == S[j] && dp[i+1][j-1] == 1){
dp[i][j] = 1;
ans = L;//更新最长回文子串长度
}
}
}
printf("%d\n",ans);
return 0;
}