其他分享
首页 > 其他分享> > Luogu P2657 [SCOI2009]windy数

Luogu P2657 [SCOI2009]windy数

作者:互联网

传送门

终于明白数位DP是什么了

虽然说是dp,实际上是记忆化搜索

看这道题的题面:

不含前导零,且相邻两个数字之差至少为2的正整数被称为windy数。

求:在A和B之间,包括A和B,总共有多少个windy数?

这道题中给出了几个限定条件。

其中,前导零、数字上限(A,B)是数位dp的题目中比较常见的。

而“相邻数字之差至少为2”则是这道题特有的限定条件。

考虑用记忆化搜索来解决这道题。

首先考虑普通的搜索。搜索两次,边界分别设为给定的两个数,那么(solve[b] - solve[A-1])即为答案。

假设已经处理出了边界条件,共有n个数位,且从小到大第i位上的数字是num[i]。

那么,搜索时需要传递哪些变量?

 

int dfs(int cur,int las,bool start,bool lim) {
    if(!cur) return 1;
    if(f[cur][las] && !start && !lim) return f[cur][las];
    int ans = 0;
    for(int i = 0; i <= (lim?num[cur]:9); i++) {
        if(!start && abs(i-las)<2) continue;
        ans += dfs(cur-1,i,(start && !i),(lim && i==num[cur]));
    }
    if(!start && !lim) f[cur][las] = ans;
    return ans;
}

 

cur表示当前是第几位(初始为n)

las表示上一位的数

start表示之前的所有数位是否为前导零

lim表示之前的所有数位是否达到边界

普通的dfs完成了,考虑记忆化搜索。

需要解决的问题是:什么样的答案是可以记下来的?

能够直接调用的答案必须是有普遍性的。

可以想到,如果确定了当前位数和上一位位数,答案就是固定的。

不过,这也存在特殊情况。如果start或者lim为true,容易举出反例。

那么,排除掉这两种情况,其余答案记录,即为正解。

代码如下

#include<cstdio>
#include<iostream>
#include<cmath>
#include<cstring>
#include<queue>
#define MogeKo qwq
using namespace std;
int A,B,n,f[20][20],num[20];


int dfs(int cur,int las,bool start,bool lim) {
    if(!cur) return 1;
    if(f[cur][las] && !start && !lim) return f[cur][las];
    int ans = 0;
    for(int i = 0; i <= (lim?num[cur]:9); i++) {
        if(!start && abs(i-las)<2) continue;
        ans += dfs(cur-1,i,(start && !i),(lim && i==num[cur]));
    }
    if(!start && !lim) f[cur][las] = ans;
    return ans;
}

int solve(int x) {
    memset(f,0,sizeof(f));
    n = 0;
    while(x) {
        num[++n] = x%10;
        x /= 10;
    }
    return dfs(n,0,1,1);
}

int main() {
    scanf("%d%d",&A,&B);
    printf("%d",solve(B)-solve(A-1));
    return 0;
}
View Code

标签:cur,int,Luogu,start,windy,lim,true,P2657,las
来源: https://www.cnblogs.com/mogeko/p/11249753.html