【BZOJ】1026 [SCOI2009]windy数

算法】数位DP

【题解】参考题解

记忆化搜索挺好写的,而且比DP+递推快。

大概思路是记录h(高度),pre(前一位数字),limit(限制)。

从根往叶子走,limit=0时,扫0~9判断符合条件就递归。

limit=1时,也就是当前位于n上,只能扫0~end-1,end就要limit=1来递归。

过程中记录一下f数组防止复算。

#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
int f[20][20],l,r,a[20];
int dfs(int h,int pre,int limit)
{
    if(h==-1)return 1;
    if(pre!=-1&&!limit&&f[h][pre]!=-1)return f[h][pre];
    int end=limit?a[h]:9;int ans=0;
    for(int i=0;i<=end;i++)
     {
         if(pre==-1&&i==0)ans+=dfs(h-1,-1,end==0?1:0);
          else
         if(abs(i-pre)>=2||pre==-1)
         {
             if(i==end&&limit)ans+=dfs(h-1,i,1);
              else ans+=dfs(h-1,i,0);
         }
     }
    if(!limit)f[h][pre]=ans;
    return ans;
}
int solve(int x)
{
    if(x==0)return 1;
    int n=0;
    while(x>0)a[n++]=x%10,x/=10;
    return dfs(n-1,-1,1);
}
int main()
{
    scanf("%d%d",&l,&r);
    memset(f,-1,sizeof(f));
    printf("%d",solve(r)-solve(l-1));
    return 0;
}

时间: 2024-08-03 11:10:22

【BZOJ】1026 [SCOI2009]windy数的相关文章

bzoj 1026 [SCOI2009]windy数(数位DP)

1026: [SCOI2009]windy数 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 4550  Solved: 2039[Submit][Status][Discuss] Description windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道,在A和B之间,包括A和B,总共有多少个windy数? Input 包含两个整数,A B. Output 一个整数. Sample

bzoj 1026 [SCOI2009]windy数 数位dp

1026: [SCOI2009]windy数 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/problem.php?id=1026 Description windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道,在A和B之间,包括A和B,总共有多少个windy数? Input 包含两个整数,A B. Output 一个整数.

bzoj 1026 [ SCOI2009 ] windy数 —— 数位DP

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1026 蛮简单的数位DP,预处理 f[i][j] 表示 i 位数,以 j 开头的 windy 数个数: 但不明白为什么最后一位拿出来特判 ret++  不对,而写在循环里,特判 i==1 就对了... 代码如下: #include<iostream> #include<cstdio> #include<cstring> #include<algorithm&g

bzoj 1026 SCOI2009 windy数

Description windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道, 在A和B之间,包括A和B,总共有多少个windy数? Input 包含两个整数,A B. Output 一个整数 Sample Input [输入样例一] 1 10 [输入样例二] 25 50 Sample Output [输出样例一] 9 [输出样例二] 20 HINT [数据规模和约定] 100%的数据,满足 1 <= A <= B <= 2

BZOJ 1026: [SCOI2009]windy数 【数位dp】

Description windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道,在A和B之间,包括A和B,总共有多少个windy数? Input 包含两个整数,A B. Output 一个整数. Sample Input [输入样例一] 1 10 [输入样例二] 25 50 Sample Output [输出样例一] 9 [输出样例二] 20 HINT [数据规模和约定] 100%的数据,满足 1 <= A <= B <= 2

bzoj 1026 [SCOI2009]windy数——数位dp水题

题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1026 迷恋上用dfs写数位dp了. #include<iostream> #include<cstdio> #include<cstring> #include<algorithm> using namespace std; const int N=15; int l,r,dg[N],dp[N][N]; int dfs(int p,int lst,bo

数位DP BZOJ 1026 [SCOI2009]windy数

题目链接 前面全是0的情况特判 #include <bits/stdc++.h> int dp[10][10]; int digit[10]; int DFS(int pos, int val, int zero, bool limit) { if (pos == -1) { return 1; } int &now = dp[pos][val]; if (!limit && zero && now != -1) { return now; } int

1026: [SCOI2009]windy数(数位dp)

1026: [SCOI2009]windy数 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 9016  Solved: 4085[Submit][Status][Discuss] Description windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道,在A和B之间,包括A和B,总共有多少个windy数? Input 包含两个整数,A B. Output 一个整数 Sample I

【BZOJ】1026: [SCOI2009]windy数(数位dp)

http://www.lydsy.com/JudgeOnline/problem.php?id=1026 我果然很弱啊... 考虑数位dp.枚举每一位,然后限制下一位即可. 一定要注意啊!在dfs的时候line这个要&&啊....要不然wa了两发.. #include <cstdio> #include <cstring> #include <cmath> #include <string> #include <iostream>

1026: [SCOI2009]windy数

Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 9670  Solved: 4445[Submit][Status][Discuss] Description windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道,在A和B之间,包括A和B,总共有多少个windy数? Input 包含两个整数,A B. Output 一个整数 Sample Input [输入样例一] 1 10 [输入样例