经典动态规划问题复习
一、最长上升子序列
#include <bits/stdc++.h>
int maxlen[10005];
int a[10005];
using namespace std;
int main()
{
int n;
cin >> n;
for(int i=1;i<=n;i++)
{
cin >> a[i];
}
for(int i=1;i<=n;i++)
{
maxlen[i] = 1;
}
for(int i=2;i<=n;i++)
{
for(int j=1;j<i;j++)
{
if(a[j]<a[i])
{
maxlen[i] = max(maxlen[i],maxlen[j]+1);
}
}
}
cout << *max_element(maxlen+1,maxlen+n+1);
}
1、思路:
选好子问题是关键。到某一个位置为止的最大上升子序列=前面位置的最大上升子序列+1。
2、max_element是用来来查询最大值所在的第一个位置。前面加 * 取内容即可得数组中最大值。
二、公共子序列
#include<bits/stdc++.h>
using namespace std;
char a[10005],b[10005];
int s[10005][10005];
int main()
{
while(cin >> a >> b)
{
int len1 = strlen(a);
int len2 = strlen(b);
for(int i=0;i<len1;i++)
{
s[i][0] = 0;
}
for(int i=0;i<len2;i++)
{
s[0][i] = 0;
}
for(int i=1;i<=len1;i++)
{
for(int j=1;j<=len2;j++)
{
if(a[i-1]==b[j-1])
{
s[i][j] = s[i-1][j-1]+1;
}
else
{
s[i][j] = max(s[i-1][j],s[i][j-1]);
}
}
}
cout << s[len1][len2] << endl;
}
}
三、切割回文串
#include<bits/stdc++.h>
using namespace std;
string s;
bool judge(int start,int ed)
{
for(int i=start;i<=(ed+start)/2;i++)
{
if(s[i]!=s[ed-(i-start)])
{
return false;
}
}
return true;
}
int main()
{
int t;
cin >> t;
while(t--)
{
cin >> s;
int len = s.length();
int dp[len+1];
for(int i=1;i<=len;i++)
{
dp[i]=i-1;
}
for(int i=1;i<=len;i++)
{
if(judge(0,i-1))//是回文串,则不需要切割
{
dp[i]=0;
continue;
}
for(int j=1;j<i;j++)
{
if(judge(j,i-1))
{
dp[i] = min(dp[i],dp[j]+1);
}
}
}
cout << dp[len] << endl;
}
}
四、字符串距离
#include<bits/stdc++.h>
#define MAX 10000
using namespace std;
int lena,lenb;
int f[MAX][MAX];
char a[MAX],b[MAX];
int solve_min(int a,int b,int c)
{
int d = min(a,b);
if(d<c) return d;
else return c;
}
void dp()
{
for(int i=1;i<=lena;i++)
{
f[i][0] = i;
}
for(int j=1;j<=lenb;j++)
{
f[0][j] = j;
}
for(int i=1;i<=lena;i++)
{
for(int j=1;j<=lenb;j++)
{
if(a[i-1]==b[j-1])
{
f[i][j] = f[i-1][j-1];
}
else f[i][j] = solve_min(f[i-1][j],f[i][j-1],f[i-1][j-1])+1;
// f[i-1][j]是删除一个,f[i][j-1]是末尾加一个,f[i-1][j-1]是替换
}
}
}
int main()
{
int t;
cin >> t;
while (t--)
{
scanf("%s %s",a,b);
lena = strlen(a);
lenb = strlen(b);
dp();
printf("%d\n",f[lena][lenb]);
}
}