HDU 2089 不要62
c a l cal cal函数(一般情况):
注意每次初始化 D P DP DP 数组为 − 1 −1 −1,长度 l e n = 0 len=0 len=0
基本参数:
l e n : len: len: 数位长度,一般根据这个来确定数组范围
a
i
:
a_i:
ai: 每个数位具体数字
返回值
r
e
t
u
r
n
return
return 根据题目的初始条件来确定前导
0
0
0 以及
p
r
e
pre
pre
DFS 函数(一般情况):
变量
r
e
s
res
res 来记录答案,初始化一般为
0
0
0
变量
u
p
up
up 表示能枚举的最高位数
采用记忆化搜索的方式:
if (!limit && !lead && dp[pos][pre] != -1) return dp[pos][pre];
只有无限制、无前导零才算,不然都是未搜索完的情况。
return limit ? res : dp[pos][pre] = res;
如果最后还有限制,那么返回 res,否则返回 dp[pos][pre]
基本参数:
假设数字 x x x 位数为 a 1 ⋯ a n a_1⋯a_n a1⋯an
必填参数:
p o s : pos: pos: 表示数字的位数
从末位或第一位开始,要根据题目的数字构造性质来选择顺序,一般选择从 a 1 a_1 a1 到 a n a_n an 的顺序。初始从 l e n len len 开始的话,边界条件应该是 p o s = 0 pos=0 pos=0,限制位数应该是 a p o s a_{pos} apos,DFS 时 p o s − 1 pos−1 pos−1 ;初始从 1 1 1 开始的话,边界条件应该是 p o s > l e n pos>len pos>len ,限制位数应该是 a l e n − p o s + 1 a_{len−pos+1} alen−pos+1, D F S DFS DFS 时 p o s + 1 pos+1 pos+1。两种都可以,看个人习惯。
l i m i t : limit: limit: 可以填数的限制(无限制的话 ( l i m i t = 0 ) (limit=0) (limit=0) 0 ∼ 9 0∼9 0∼9 随便填,否则只能填到 a i a_i ai)
如果搜索到 a 1 ⋯ a p o s ⋯ a n a_1\cdots a_{pos} \cdots a_n a1⋯apos⋯an,原数位为 a 1 ⋯ a k ⋯ a n a_1\cdots a_k \cdots a_n a1⋯ak⋯an,那么我们必须对接下来搜索的数加以限制,也就是不能超过区间右端点 R R R,所以要引入 l i m i t limit limit 这个参数,如果 l i m i t = 1 limit=1 limit=1,那么最高位数 u p ≤ a p o s + 1 up \le a_{pos+1} up≤apos+1,如果没有限制,那么 u p = 9 up=9 up=9(十进制下)这也就是确定搜索位数上界的语句
limit ? a[pos] : 9
;
如果 l i m i t = 1 limit=1 limit=1 且已经取到了能取到的最高位时 ( a p o s = a k ) (a_{pos}=a_k) (apos=ak),那么下一个 l i m i t = 1 limit=1 limit=1
如果 l i m i t = 1 limit=1 limit=1 且没有取到能取到的最高位时 ( a p o s < a k ) (a_{pos} <a_k) (apos<ak),那么下一个 l i m i t = 0 limit=0 limit=0
如果 l i m i t = 0 limit=0 limit=0,那么下一个 l i m i t = 0 limit=0 limit=0,因为前一位没有限制后一位必定没有限制。
所以我们可以把这 3 3 3 种情况合成一个语句进行下一次搜索:limit && i == up
( i (i (i为当前枚举的数字 ) ) )
可选参数:
p r e : pre: pre: 表示上一个数是多少
有些题目会用到前面的数
l e a d : lead: lead: 前导零是否存在, l e a d = 1 lead=1 lead=1 存在前导零,否则不存在。
一般来说有些题目不加限制前导零会影响数字结构,所以 l e a d lead lead 是一个很重要的参数。
如果 l e a d = 1 lead=1 lead=1 且当前位为 0 0 0,那么说明当前位是前导 0 0 0,继续搜索 p o s + 1 pos+1 pos+1,其他条件不变。
如果 l e a d = 1 lead=1 lead=1 且当前位不为 0 0 0,那么说明当前位是最高位,继续搜索 p o s + 1 pos+1 pos+1,条件变动。
如果 l e a d = 0 lead=0 lead=0,则不需要操作。
s u m : sum: sum: 搜索到当前所有数字之和
有些题目会出现数字之和的条件
c n t : cnt: cnt: 某个数字出现的次数
有些题目会出现某个数字出现次数的条件
参数基本的差不多这些,有些较难题目会用到更多方法或改变 DP \text{DP} DP状态
#include <bits/stdc++.h>
using namespace std;
const int N = 11;
int l, r, len, a[N], dp[N][N];
int dfs(int pos, int pre, int limit) {
if (!pos) return 1;
if (!limit && dp[pos][pre] != -1) return dp[pos][pre];
int res = 0, up = limit ? a[pos] : 9;
for (int i = 0; i <= up; i ++) {
if (i == 4 || (i == 2 && pre == 6)) continue;
res += dfs(pos - 1, i, limit && i == up);
}
return limit ? res : dp[pos][pre] = res;
}
int cal(int x) {
memset(dp, -1, sizeof dp);
len = 0;
while (x) a[++ len] = x % 10, x /= 10;
return dfs(len, 0, 1);
}
int main() {
while (cin >> l >> r, l || r) {
cout << cal(r) - cal(l - 1) << endl;
}
return 0;
}