题面

我们定义一个数是单调数,当且仅当构成这个数每一个数位都是单调不降或不增的。

例如 \(123\) 和 \(321\) 和 \(221\) 和 \(111\) 是单调的,而 \(312\) 不是单调的。

给定 \(T\) 组 \(l, r\),每次询问 \([l, r]\) 中有几个单调的数。

\(l, r \le 10 ^ {18}, T \le 10^4\)

题解

今天 hihoCoder 竟然 \(AK\) 了,真舒服(虽然题目很水,还被罚时坑惨了)qwq

显然考虑一个数位 \(dp\) ,不难发现我们只需要记下当前在哪一位,以及最后一位是什么数字就行了。

然后对于不降和不增做个两遍就行了,但有很多细节后面细讲。

写的时候发现自己摸索了一套数位 \(dp\) 的套路?(逃

套路:

我们常常是要求 \(\le n\) 的有多少个满足要求的数。

这个限制有些恶心,我们需要多一位来看是否被限制。

我们一般按位考虑,令 \(dp[i][0 / 1]\) 为到从高到低考虑到第 \(i\) 位,当前有/没被 \(n\) 限制。

我们考虑把 \(n\) 按位拆下来,变成一个数组 lim[i] ,然后取出 \(n\) 的位数 Bit

每次考虑后一位放什么数字就行了。具体实现如下(用刷表法方便转移):

dp[0][0] = 1;
for (int i = 0; i < Bit; ++ i)
for (int now = 0; now <= 1; ++ now)
for (int dig = 0; dig <= 9; ++ dig) if (now || (dig <= lim[i + 1]))
dp[i + 1][now || (dig < lim[i + 1])] += dp[i][now];

然后最后的答案就是

ans = dp[Bit][0] + dp[Bit][1];

不难发现这样写,每个位数的数都会被考虑到。因为我们枚举的时候允许了前缀 \(0\) 的存在。

并且如果存在前缀 \(0\) 那么后面的所有数都不会存在限制了,可以随便填。

但是注意这样的话,全部为 \(0\) 的也会考虑进去,我们平常要考虑是否 \(-1\) 就行了。

对于这道题我们可以类比这种方法去做。

首先把答案差分表示 \(ans = ans_r - ans_{l - 1}\) 。

然后令 dp[i][j][0/1] 为到第 \(i\) 位,最后一次填 \(j\) ,有/没 被 \(n\) 限制住的情况。

直接这样写的话,递增是没有问题的,递减的时候就会存在问题了。

因为我们把前导 \(0\) 考虑进去了,结果导致没有正确算上这部分贡献。

所以我们还要多一维,也就是 dp[i][j][0/1][0/1] 前三个同上,最后一个意义是当前完全不/是前缀 \(0\) 。

然后转移的时候也是枚举数字,然后按照两种情况考虑下填的这个数的限制就行了。

注意有一些数会被递增递减算两次,也就是 \(111\) ,\(3333\) ,这些完全相同的数。可以直接暴力枚举减去就行了。

复杂度是 \(O(T * 18 * 10)\) 的。

总结

碰到数位 \(dp\) 直接上套路去讨论。

然后就需要对于具体问题具体分析了,根据题目要求设出需要的状态。

有时候可以根据需要卡卡状态数,因为通常不可能到满。

一定要写个暴力拍,这个东西其实很好调?

代码

建议看看代码,其实写的很简洁?(可读性也是很鲁棒的qwq)

#include <bits/stdc++.h>

#define For(i, l, r) for(register int i = (l), i##end = (int)(r); i <= i##end; ++i)
#define Fordown(i, r, l) for(register int i = (r), i##end = (int)(l); i >= i##end; --i)
#define Set(a, v) memset(a, v, sizeof(a))
#define Cpy(a, b) memcpy(a, b, sizeof(a))
#define debug(x) cout << #x << ": " << x << endl
#define DEBUG(...) fprintf(stderr, __VA_ARGS__) using namespace std; inline bool chkmin(int &a, int b) {return b < a ? a = b, 1 : 0;}
inline bool chkmax(int &a, int b) {return b > a ? a = b, 1 : 0;} inline int read() {
int x = 0, fh = 1; char ch = getchar();
for (; !isdigit(ch); ch = getchar()) if (ch == '-') fh = -1;
for (; isdigit(ch); ch = getchar()) x = (x << 1) + (x << 3) + (ch ^ 48);
return x * fh;
} void File() {
#ifdef zjp_shadow
freopen ("1770.in", "r", stdin);
freopen ("1770.out", "w", stdout);
#endif
} typedef long long ll;
ll dp[19][10][2][2]; int lim[19];
inline int Get_Bit(ll x) {
int tot = 0;
for (; x; x /= 10) lim[++ tot] = x % 10;
reverse(lim + 1, lim + tot + 1);
return tot;
} inline ll Calc(ll n) { if (!n) return 0; ll ans = 0; For (opt, 0, 1) {
Set(dp, 0); dp[0][0][0][1] = 1; int Bit = Get_Bit(n);
For (i, 0, Bit - 1) For (j, 0, 9) For(now, 0, 1) For (fir, 0, 1) {
For (dig, opt ? 0 : j, opt ? (fir ? 9 : j) : 9) if (now || dig <= lim[i + 1])
dp[i + 1][dig][now || (dig < lim[i + 1])][fir && !dig] += dp[i][j][now][fir];
} For (j, 0, 9) For(now, 0, 1) For (fir, 0, 1) ans += dp[Bit][j][now][fir]; -- ans;
} For (dig, 1, 9) {
ll tmp = dig;
while (tmp <= n) -- ans, tmp = tmp * 10 + dig;
} return ans; } int main() { File(); for (int cases = read(); cases; -- cases) {
ll l, r; cin >> l >> r;
printf ("%lld\n", Calc(r) - Calc(l - 1));
} return 0; }

hihoCoder #1770 : 单调数(数位dp)的更多相关文章

  1. 【BZOJ1662】[Usaco2006 Nov]Round Numbers 圆环数 数位DP

    [BZOJ1662][Usaco2006 Nov]Round Numbers 圆环数 Description 正如你所知,奶牛们没有手指以至于不能玩"石头剪刀布"来任意地决定例如谁 ...

  2. 【BZOJ-1026】windy数 数位DP

    1026: [SCOI2009]windy数 Time Limit: 1 Sec  Memory Limit: 162 MBSubmit: 5230  Solved: 2353[Submit][Sta ...

  3. CCF 201312-4 有趣的数 (数位DP, 状压DP, 组合数学+暴力枚举, 推公式, 矩阵快速幂)

    问题描述 我们把一个数称为有趣的,当且仅当: 1. 它的数字只包含0, 1, 2, 3,且这四个数字都出现过至少一次. 2. 所有的0都出现在所有的1之前,而所有的2都出现在所有的3之前. 3. 最高 ...

  4. bzoj 1026 [SCOI2009]windy数 数位dp

    1026: [SCOI2009]windy数 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline ...

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

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

  6. 【BZOJ 3326】[Scoi2013]数数 数位dp+矩阵乘法优化

    挺好的数位dp……先说一下我个人的做法:经过观察,发现这题按照以往的思路从后往前递增,不怎么好推,然后我就大胆猜想,从前往后推,发现很好推啊,维护四个变量,从开始位置到现在有了i个数 f[i]:所有数 ...

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

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

  8. 【bzoj1026】[SCOI2009]windy数 数位dp

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

  9. [bzoj1026][SCOI2009]windy数——数位dp

    题目 求[a,b]中的windy数个数. windy数指的是任意相邻两个数位上的数至少相差2的数,比如135是,134不是. 题解 感觉这个题比刚才做的那个简单多了...这个才真的应该是数位dp入门题 ...

随机推荐

  1. 【阿里云】在 Windows Server 2016 下使用 FileZilla Server 安装搭建 FTP 服务

     Windows Server 2016 下使用 FileZilla Server 安装搭建 FTP 服务 一.安装 Filezilla Server 下载最新版本的 Filezilla Server ...

  2. nodejs+express+mongodb写api接口的简单尝试

    1:启动mongodb服务 我的mongoDB的安装目录:E:\mongoDB\bin,版本:3.4.9 打开cmd  -> e:(进入e盘) -> cd mongoDB/bin(进入mo ...

  3. 设计模式系列之策略模式(Strategy Pattern)

    意图:定义一系列的算法,把它们一个个封装起来, 并且使它们可相互替换. 主要解决:在有多种算法相似的情况下,使用 if...else 所带来的复杂和难以维护. 何时使用:一个系统有许多许多类,而区分它 ...

  4. js计算剩余分钟

    // 剩余时间提醒 function checkTime() { if (timeCompare()) { document.getElementById('distanceDeadline').in ...

  5. 会话固定攻击 - yxcms session固定漏洞

    目录 会话固定攻击 e.g. yxcms session固定攻击 分析 了解更多 会话固定攻击 Session fixation attack(会话固定攻击)是利用服务器的session不变机制,借他 ...

  6. Python第八天 模块 包 全局变量和内置变量__name__ Python path

    Python第八天  模块   包   全局变量和内置变量__name__    Python path 目录 Pycharm使用技巧(转载) Python第一天  安装  shell  文件 Pyt ...

  7. eclipse设置新建jsp默认编码格式utf-8

  8. String输出结果to thi

    http://blog.csdn.net/itmyhome1990/article/details/9132929

  9. 利用BLKTRACE分析IO性能

    在Linux系统上,如果I/O发生性能问题,有没有办法进一步定位故障位置呢?iostat等最常用的工具肯定是指望不上的,[容易被误读的iostat]一文中解释过await表示单个I/O所需的平均时间, ...

  10. vue 路由变化页面数据不刷新问题(缓存)

    每天记录一点点,把我遇到的问题记录下来, 希望可以帮助到更多和我遇到同样问题的人. 问题描述:通过调接口,动态显示帮助页面的问题列表, 问题列表有多级,当点击的这个问题没有下一级问题的时候跳入内容页. ...