[hdu 2089] 不要62 数位dp|dfs 入门

题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=2089

题意:求[n, m]区间内不含4和62的数字个数。

这题有两种思路,直接数位dp和dfs

数位dp:

  dp[i][j]表示i位数,首位是j的符合要求的数字个数。

  j = 4时    dp[i][j] = 0

  j != 4时

   

  例如求dp[3][2],2xx的个数,2已经确定了,2后面xx的个数即为2xx的个数,只用求出0x, 1x, 2x...9x的个数之和即可。同时要注意限制条件,dp[i][4]均为0,如果i位首位为6,i-1位首位为2的话也为0。这样我们首先预处理下,然后由此可以求区间内所符合要求的数字个数。

  以求[0, 365]为例,先求0xx, 1xx, 2xx, xx的个数即为每个的个数,当然如果是555的的话4xx是跳过的,或者前一位是6,那么2xx也要跳过。然后求[300, 365]的个数,已经确定首位为3,求3xx的个数,然后类似的确定xx的个数。如果遇到456这种情况,只用求0xx,1xx,2xx,3xx的个数,4xx就不用求了,因此最外层循环就可以停止了。类似的6223.. 没必要求[6200,6223]了。

代码:

#include <iostream>
#include <stdio.h>
#include <cstring>
#include <algorithm>
using namespace std;

int dp[10][10];
int d[10];

void init()
{
    memset(dp, 0, sizeof(dp));
    dp[0][0] = 1;
    for(int i = 1; i <= 7; i++)
        for (int j = 0; j <= 9; j++)
            for (int k = 0; k <= 9; k++)
                if (j != 4 && !(j==6 && k==2))
                    dp[i][j] += dp[i-1][k];
}

int solve(int n)
{
    int ans = 0, len = 0;
    while (n) {
        d[++len] = n % 10;
        n /= 10;
    }
    d[len+1] = 0;
    for (int i = len; i >= 1; i--) {
        for (int j = 0; j < d[i]; j++) {
            if (d[i+1] != 6 || j != 2)
                ans += dp[i][j];
        }
        if (d[i]==4 || (d[i+1]==6 && d[i]==2))
            break;
    }
    return ans;
}


int main()
{
    freopen("1.txt", "r", stdin);
    int n, m;
    init();
    while (~scanf("%d%d", &n, &m)) {
        if (n + m == 0) break;
        printf("%d
", solve(m+1)-solve(n));
    }


    return 0;
}

DFS+记忆化搜索:

  dp[i][j]表示i位数,前一位数组是否为6的符合要求的个数。

  dfs的参数l是当前的位数,从最高位开始搜索。six是前一位是否为6,limit是最高位是否受限,如365,最高位就受限与0~3,然后开始搜索0xx, 1xx, 2xx, 3xx, 其中0xx,1xx,2xx中的xx都是不受限的,0~99均可取,而3xx中的xx要受65的限制,搜索下一位时继续设限。如果该位为4,或上一位为6,该位为2时就跳过不搜。另外搜索中有大量重复,所以采用记忆化搜索。如果受限的话就不能采用记忆化搜索的结果。

代码:

#include <iostream>
#include <stdio.h>
#include <cstring>
#include <algorithm>
using namespace std;

int digit[10], dp[10][2], v[10][2];

int dfs(int l, bool six, bool limit)
{
    if (l == 0) return 1;
    if (!limit && v[l][six]) return dp[l][six];
    int len = limit ? digit[l] : 9;
    int nx = 0;
    for (int i = 0; i <= len; i++) {
        if ((i == 4) || (six&&i==2))
            continue;
        nx += dfs(l-1, i==6, limit&&(i==len));
    }
    if (!limit) {
        v[l][six] = true;
        dp[l][six] = nx;
    }
    return nx;
}

int sum(int n)
{
    memset(dp, 0, sizeof(dp));
    memset(v, 0, sizeof(v));
    int pos = 0;
    while (n) {
        digit[++pos] = n % 10;
        n /= 10;
    }
    int ans = dfs(pos, false, true);
    return ans;
}
int main()
{
    //freopen("1.txt", "r", stdin);
    int n, m;
    while (~scanf("%d%d", &n, &m)) {
        if (n + m == 0) break;
        printf("%d
", sum(m)-sum(n-1));
    }

    return 0;
}

ps:注意两者在[n, m]时的处理。

  

  

原文地址:https://www.cnblogs.com/whileskies/p/7325758.html