不要62

数位(dp)记忆化搜索

题面链接

不要62

不吉利的数字为所有含有 (4)(62) 的号码。例如:(62315,73418,88914) 都属于不吉利号码。但是,(61152) 虽然含有 (6)(2),但不是 连号,所以不属于不吉利数字之列。

你的任务是,对于每次给出的一个牌照号区间 ([n,m]),推断出交管局今后又要实际上给多少辆新的士车上牌照了。

输入格式

输入包含多组测试数据,每组数据占一行。

每组数据包含一个整数对 (n)(m)

当输入一行为(“0 0”)时,表示输入结束。

输出格式

对于每个整数对,输出一个不含有不吉利数字的统计个数,该数值占一行位置。

数据范围

(1≤n≤m≤10^9)

输入样例:

1 100
0 0

输出样例:

80

思路

我们可以发现我们的答案其实只和我们相邻的两个数字有关系

所以我们的(dfs)就记录两个信息,当前位置(pos),前一个位置(pre)

我们肯定要维护的两个信息就是(limit)是否顶头和(lead)是否有前导零

我们可以发现在这个代码里面,我们的前导零判和没判的答案其实是一样的

也就是这个位置

if(i==4||(i==2&&pre==6))continue;

在这个题里面前导零并不会影响我们的答案,举个例子

当我们有个数是(04)的时候,虽然我们没有判掉前导零,但是我们在后续的部分(i==4)的时候,我们还是

会判掉他,再比如(062)它还是有前导零,但是它还是会在后续的(i==2 并且 pre==6)判掉

所以也是不会影响我们的答案的

但是还是判断了的好,因为答案肯定不会错

核心代码

int dfs(int pos,int pre,int limit,int lead)
{
	if(pos==0)return 1;//如果已经枚举完了最后每一位,直接返回
    if(!limit&&!lead&&dp[pos][pre]!=-1)	return dp[pos][pre];
    int bound=(limit==1)?a[pos]:9;
    long long ans=0;
    for(int i=0;i<=bound;i++)
    {
        if(i==4||(i==2&&pre==6))continue;
        ans+=dfs(pos-1,i,limit&&(i==bound),lead&&(i==0));
    }
	if(!limit&&!lead) dp[pos][pre]=ans;
    return ans;
}
原文地址:https://www.cnblogs.com/bangdexuanyuan/p/13965403.html