一道比较基础的数位DP,还是挺套路的。

首先看题,发现这个性质和数的大小无关,因此我们可以直接数位DP,经典起手式:

\(f[a,b]=f(b)-f(a-1)\)

然后考虑如何求解\(f(x)\)。我们首先可以在不考虑数的大小的情况下得出长为\(i\)位且以数字\(j\)开头的windy数字个数。

这个还是很好求的,我们设\(f_{i,j}\),然后每一位从上一位转移即可。

然后考虑如何统计,我们把要统计的数分成三类:

  1. 位数比原来的数小的,且开头不能为\(0\)的数的总数。这个直接累加即可。
  2. 位数和原来的数一样,但开头的数字比原来的数字小的数的总数。由于这样后面也可以随便取,因此累加即可。
  3. 位数和原来的数一样,且开头的数字也一样的数的总数。这个就比较难求了。我们考虑枚举后面的每一位,在不填到最高位的情况下都可以继续累加。然后我们假定这一位也到达了最高位,然后继续统计即可。

注意在统计第三种数时要注意若此时相邻的两个数的最高位只差小于\(2\)需要直接退出

最后我们发现这个只能统计所以小于\(x\)的windy数。因此我们把原来的起手式变成\(f(b+1)-f(a)\)即可。

CODE

#include<cstdio>
#include<cstring>
using namespace std;
int a,b,bit[15],cnt;
long long f[15][10];
inline int abs(int x)
{
return x>0?x:-x;
}
inline void init(void)
{
register int i,j,k;
for (i=0;i<=9;++i)
f[1][i]=1;
for (i=2;i<=10;++i)
for (j=0;j<=9;++j)
for (k=0;k<=9;++k)
if (abs(j-k)>=2) f[i][j]+=f[i-1][k];
}
inline void solve(int x)
{
while (x) bit[++cnt]=x%10,x/=10;
}
inline long long get(int x)
{
register int i,j,k,w; long long ans=0;
cnt=0; solve(x);
for (i=1;i<cnt;++i)
for (j=1;j<=9;++j)
ans+=f[i][j];
for (i=1;i<bit[cnt];++i)
ans+=f[cnt][i];
for (i=cnt-1;i>=1;--i)
{
for (j=0;j<bit[i];++j)
if (abs(j-bit[i+1])>=2) ans+=f[i][j];
if (abs(bit[i]-bit[i+1])<2) break;
}
return ans;
}
int main()
{
scanf("%d%d",&a,&b); init();
printf("%lld",get(b+1)-get(a));
return 0;
}

Luogu P2657 [SCOI2009]windy数的更多相关文章

  1. luogu P2657 [SCOI2009]windy数 数位dp 记忆化搜索

    题目链接 luogu P2657 [SCOI2009]windy数 题解 我有了一种所有数位dp都能用记忆话搜索水的错觉 代码 #include<cstdio> #include<a ...

  2. 题解 BZOJ1026 & luogu P2657 [SCOI2009]windy数 数位DP

    BZOJ & luogu 看到某大佬AC,本蒟蒻也决定学习一下玄学的数位$dp$ (以上是今年3月写的话(叫我鸽神$qwq$)) 思路:数位$DP$ 提交:2次 题解:(见代码) #inclu ...

  3. P2657 [SCOI2009]windy数

    P2657 [SCOI2009]windy数 题目描述 windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道, 在A和B之间,包括A和B ...

  4. 洛谷 P2657 [SCOI2009]windy数 解题报告

    P2657 [SCOI2009]windy数 题目描述 \(\tt{windy}\)定义了一种\(\tt{windy}\)数.不含前导零且相邻两个数字之差至少为\(2\)的正整数被称为\(\tt{wi ...

  5. 洛谷——P2657 [SCOI2009]windy数

    P2657 [SCOI2009]windy数 题目大意: windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道, 在A和B之间,包括A和 ...

  6. C++ 洛谷 P2657 [SCOI2009]windy数 题解

    P2657 [SCOI2009]windy数 同步数位DP 这题还是很简单的啦(差点没做出来 个位打表大佬请离开(包括记搜),我这里讲的是DP!!! 首先Cal(b+1)-Cal(a),大家都懂吧(算 ...

  7. 洛谷P2657 [SCOI2009]windy数 [数位DP,记忆化搜索]

    题目传送门 windy数 题目描述 windy定义了一种windy数.不含前导零且相邻两个数字之差至少为2的正整数被称为windy数. windy想知道, 在A和B之间,包括A和B,总共有多少个win ...

  8. [洛谷P2657][SCOI2009]windy数

    题目大意:不含前导零且相邻两个数字之差至少为$2$的正整数被称为$windy$数.问$[A, B]$内有多少个$windy$数? 题解:$f_{i, j}$表示数有$i$位,最高位为$j$(可能为$0 ...

  9. P2657 [SCOI2009]windy数 数位dp

    数位dp之前完全没接触过,所以NOIP之前搞一下.数位dp就是一种dp,emm……用来求解区间[L,R]内满足某个性质的数的个数,且这个性质与数的大小无关. 在这道题中,dp[i][j]代表考虑了i位 ...

随机推荐

  1. OkHttp3源码详解(二) 整体流程

    1.简单使用 同步: @Override public Response execute() throws IOException { synchronized (this) { if (execut ...

  2. 逻辑回归&线性回归

    # coding:utf-8 import numpy as np from sklearn import linear_model, datasets import matplotlib.pyplo ...

  3. Node.js中文乱码解决方法

  4. datatable 列名重新排序

    1. DataTable.Columns["MONEY"].SetOrdinal[4]; 2.dt = dt.DefaultView.ToTable(false, listarr. ...

  5. mysqld_safe启动服务器总结

    mysqld_safe是服务端工具,用于启动mysqld,并且是mysqld的守护进程,mysqld_safe加&在后台运行$BASEDIR/bin/mysqld_safe & 优点就 ...

  6. python 从外部获取传入的参数

    有时候我们在执行python程序的时需要接收到外部传入的参数 python的 sys.argv[]就能实现 # test.py import sys #引入模块 str = sys.argv[1]pr ...

  7. ubuntu 14.04安装zabbix3.0以及汉化

    文章出处借鉴于 http://www.cnblogs.com/-10086/p/5317524.html 1.下载deb # wget http://repo.zabbix.com/zabbix/3. ...

  8. Python作业第一课

    零基础开始学习,最近周边的同学们都在学习,我也来试试,嘿嘿,都写下来,下次不记得了还能来看看~~ Python作业第一课1)登陆,三次输入锁定,下次不允许登陆2)设计一个三级菜单,菜单内容可自行定义, ...

  9. Eclipse更新maven项目仓库依赖

    ALT+F5 弹出 选择需要更新的项目, 点击ok, 就开始下载更新依赖的jar包了

  10. 实用的php清除html,php去除空格与换行,php清除空白行和换行,提取页面纯文本

    实用的php清除html,换行,空格类,php去除空格与换行,php清除空白行和换行,提取页面纯文本内容 方法一: function DeleteHtml($str) { $str = trim($s ...