[agc011e]Increasing Numbers

前言

1min了我还不会一点思路都没有这正常?
orz题解。

题意

把大整数n拆成若干个上升数之和,最小化上升数的个数。
上升数即从最高位往最低位是一个不下降的序列。

做法

发现任意上升数能表示成全1数的和。
如果你允许0的存在,那么每个上升数都能被拆成9个全1数的和。
全1数的位数假如是k,可以表示成 10k+119
刚好0是-1位,代入该式子也成立。
现在求一个最小的k使得
9ki=110a[i]+119=n
a[i]表示第i个全1数的位数,-1表示0。
干脆全部+1,新的a[i]表示全1数的位数+1,0表示0。
那么 9ki=110a[i]=9n+9k
这式子啥意思啊?
9n+9k的数位和等于9k!
那么我们不妨枚举k,暴力进位的复杂度是均摊o(1)的。
容易发现k达到o(n)级别以上就不可能再令式子取等号。
题解所说的二分正确性我持怀疑态度,因为我觉得数位和这玩意不单调。

#include<cstdio>
#include<algorithm>
#include<cstring>
#define fo(i,a,b) for(i=a;i<=b;i++)
using namespace std;
const int maxn=500000+10;
char s[maxn];
int a[maxn];
int i,j,k,l,t,n,m,num;
int main(){
    scanf("%s",s);
    n=strlen(s);
    reverse(s,s+n);
    fo(i,0,n-1) a[i]=s[i]-'0';
    t=i=0;
    while (i<n){
        t+=a[i]*9;
        a[i]=t%10;
        t/=10;
        if (t&&i==n-1) n++;
        i++;
    }
    fo(i,0,n-1) num+=a[i];
    fo(k,1,n*20){
        a[0]+=9;
        num+=9;
        //if (a[0]<10) continue;
        t=i=0;
        while (i<n){
            t+=a[i];
            num-=a[i];
            a[i]=t%10;
            num+=a[i];
            t/=10;
            if (!t) break;
            if (i==n-1) n++;
            i++;
        }
        /*l=(k-1)/9+1;
        if (!a[0]&&num>=k&&num<=l*9) break;*/
        if (num<=9*k) break;
    }
    printf("%d\n",k);
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值