• 不要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;
    }
    
  • 相关阅读:
    第三周作业
    第二周作业
    第一周作业
    第三次作业
    第二次作业
    c语言最后一次作业
    第14、15教学周作业
    第七周作业
    第六周随笔
    第四周作业
  • 原文地址:https://www.cnblogs.com/bangdexuanyuan/p/13965403.html
Copyright © 2020-2023  润新知