不要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;
}
posted @ 2020-11-12 18:23  邦的轩辕  阅读(311)  评论(0)    收藏  举报