Bomb 【数位dp】

The counter-terrorists found a time bomb in the dust. But this time the terrorists improve on the time bomb. The number sequence of the time bomb counts from 1 to N. If the current number sequence includes the sub-sequence "49", the power of the blast would add one point. Now the counter-terrorist knows the number N. They want to know the final points of the power. Can you help them? InputThe first line of input consists of an integer T (1 <= T <= 10000), indicating the number of test cases. For each test case, there will be an integer N (1 <= N <= 2^63-1) as the description. The input terminates by end of file marker. OutputFor each test case, output an integer indicating the final points of the power.Sample Input3150500Sample Output0115HintFrom 1 to 500, the numbers that include the sub-sequence "49" are "49","149","249","349","449","490","491","492","493","494","495","496","497","498","499",so the answer is 15.

题意 :在【0,n】范围内有出现了多少次 “49”

思路:数位dp

dp[i][0] 表示 长度为 i 不出现 “49” 的个数

dp[i][1] 表示 长度为 i 最高位为 “9” 且不含“49”的个数

dp[i][2] 表示 长度为 i 出现 “49” 的个数

 

那么对0 ~ 2^63-1 内进行初始化

ll dp[25][3];
void init(){
    dp[0][0]=1;
    dp[0][1]=dp[0][2]=0;
    for (int i=1;i<=20;i++){
        dp[i][0]=dp[i-1][0]*10-dp[i-1][1];  /* 长度为 i 中 不包含 “49”的个数
为长度为 i-1 中不含“49”的个数在第 i 位 加 0~9 的个数 减掉 当第 i 位为“4”时,长度为 i-1 
的数中最高位为“9”的个数 */
        dp[i][1]=dp[i-1][0];/*长度为 i-1的数中不含“49”的数在第 i 位加“9” 
即为dp[i][1];*/ 
        dp[i][2]=dp[i-1][2]*10+dp[i-1][1];/* 长度为i的数中含“49”的个数等于长度为 i-1 中含“49” 的数字 在第 i 位上放0~9,加当第 i 位为“4”时,长度为 i-1 的数字中,最高位
为“9”的数字。*/
    }

ll getCnt(ll x){
    ++x; /*为啥加一我也不知道,反正加一就过了 qwq */
    ll ans=len=0;
    while(x){
        num[++len]=x%10;
        x/=10;
    }
    bool fg=false;   /* 种一个标记判断到第i位是 最高位到第i+1位 是否出现“49”*/ 
    num[len+1]=-1;
    for (int i=len;i>=1;i--){
        ans+=dp[i-1][2]*num[i];  //第i位 取(0~num[i]-1)时 长度为i-1的数中含“49”的数 
        if(fg)   ans+=dp[i-1][0]*num[i]; /*如果高位中已经出现过“49”,第i位取(0~num[i]-1)时,长度为i-1中不含“49”的个数*/
        if(!fg && num[i]>4) ans+=dp[i-1][1];  /*当未标记且该位大于4时,取长度为i-1的数中最高位为“9”的数*/
        if(num[i+1]==4 && num[i]==9) fg=1; /*如果该位为9前一位为4 则标记*/
    }
    return ans;
}
过题代码如下:
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <functional>
#include <iostream>
using namespace std;
typedef long long ll;
ll x,len;
int num[25];
ll dp[25][3];
void init(){
    dp[0][0]=1;
    dp[0][1]=dp[0][2]=0;
    for (int i=1;i<=20;i++){
        dp[i][0]=dp[i-1][0]*10-dp[i-1][1];
        dp[i][1]=dp[i-1][0];
        dp[i][2]=dp[i-1][2]*10+dp[i-1][1];
    }
}
ll getCnt(ll x){
    ++x;
    ll ans=len=0;
    while(x){
        num[++len]=x%10;
        x/=10;
    }
    bool fg=false;
    num[len+1]=-1;
    for (int i=len;i>=1;i--){
        ans+=dp[i-1][2]*num[i];
        if(fg)   ans+=dp[i-1][0]*num[i];
        if(!fg && num[i]>4) ans+=dp[i-1][1];
        if(num[i+1]==4 && num[i]==9) fg=1;
    }
    return ans;
}
int main(){
    init();
    int T;
    scanf("%d",&T);
    while(T--){
        scanf("%I64d",&x);
        printf("%I64d\n",getCnt(x));
    }
    return 0;
}



posted @ 2018-05-27 22:33  Acerkoo  阅读(162)  评论(0编辑  收藏  举报