洛谷 1415

      这道题比较神奇,第一次做到这样的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;	
}


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值