这道题比较神奇,第一次做到这样的DP题,我们令dp[i]表示序列1--i中满足序列递增且使最后一个数最小,最后一个数的起始位置,这样的话转移方程为dp[i]=max(j),其中dp[j-1]--j-1这段区间的值小于j--i这段区间的值,有了这个以后我们可以知道最后一个数是什么(暂时不考虑前导0),那么我们可以再用同样的方法,令f[i]表示i--dp[n]-1这段区间里满足序列递增且使第一个数最大,第一个数的终止位置,那么我们有f[i]=max(j),其中i到j的值小于j+1到f[j+1]的值。如果不考虑前导0,我们就解决了这个问题,但是存在前导0的情况,怎么办呢?,我们可以令前导0的位置的f值为n,前导0和前面的数构成别的数,一定不会用到它的f值,如果没有,就意味这这个位置是最小数的前导0。
#include<cstdio>
#include<cstring>
#include<algorithm>
using namespace std;
#define maxn 505
int dp[maxn],n,m,f[maxn];
char s[maxn];
bool cmp(int l1,int r1,int l2,int r2)
{
while (s[l1]=='0'&&l1<r1) l1++;
while (s[l2]=='0'&&l2<r2) l2++;
if (r1-l1+1<r2-l2+1) return 1;
if (r1-l1+1>r2-l2+1) return 0;
for (int i=0;i<=r1-l1;i++)
if (s[i+l1]<s[i+l2]) return 1;
else if (s[i+l1]>s[i+l2]) return 0;
return 0;
}
int main()
{
scanf("%s",s+1);
n=strlen(s+1);
for (int i=1;i<=n;i++) dp[i]=1;
for (int i=2;i<=n;i++)
for (int j=2;j<=i;j++)
if (cmp(dp[j-1],j-1,j,i)) dp[i]=max(j,dp[i]);
int tt=dp[n];
while (s[tt-1]=='0') {tt--;f[tt]=n;}
f[dp[n]]=n;
m=tt-1;
for (int i=m;i>=1;i--)
if (cmp(i,dp[n]-1,dp[n],n)) f[i]=dp[n]-1;
for (int i=m;i>=1;i--)
for (int j=i;j<dp[n]-1;j++)
if (cmp(i,j,j+1,f[j+1]))
f[i]=max(j,f[i]);
int temp=f[1],st=1;
while (1)
{
for (int i=st;i<=temp;i++) printf("%c",s[i]);
st=temp+1;
temp=f[st];
if (st>n) break;
printf(",");
}
printf("\n");
return 0;
}