造数字(数位DP)
题面
JZM 想要创造两个数字 x 和 y,它们需要满足 x or y = T,且Lx ≤ x ≤ Rx, Ly ≤ y ≤ Ry,JZM 想知道 x and y 有多少种可能的不同的取值。若有多组 (x, y) 有相同的 x and y 的取值,则只算一次。
0 ≤ T,Lx,Rx,Ly,Ry < 260.
Sample Input
11 3 10 8 13
Sample Output
7
题解
看上去貌似是数位 DP,于是我们就设计一个 DP:
d
p
[
i
]
[
s
1
]
[
s
2
]
[
s
3
]
[
s
4
]
dp[i][s_1][s_2][s_3][s_4]
dp[i][s1][s2][s3][s4] 表示[当前转移到第
i
i
i 位,
x
x
x 是/否达到下界,
x
x
x 是/否达到上界,
y
y
y 是/否达到下界,
y
y
y 是/否达到上界]时
x and y
\text{x and y}
x and y 有多少种不同的取值。
然后观察样例,我们发现这不好转移,原因是本质相同的一种与值很容易计算多次。我们如果能保证当前位与值为 1 时和为 0 时两种情况能唯一转移到其它状态,这样就能保证不重复。但是如果当前位与值为 0 ,
x
,
y
x,y
x,y 可以是
0 & 1
\text{0 \& 1}
0 & 1,
1 & 0
\text{1 \& 0}
1 & 0 或
0 & 0
\text{0 \& 0}
0 & 0,而三种情况
{
s
1
,
s
2
,
s
3
,
s
4
}
\{s_1,s_2,s_3,s_4\}
{s1,s2,s3,s4} 结果不一样,那么就不可避免的要么计算重复,要么计算缺失了。
如果说这种情况能唯一对应一个状态的话,那就是状态集
{
{
s
1
,
s
2
,
s
3
,
s
4
}
1
,
{
s
1
,
s
2
,
s
3
,
s
4
}
2
,
.
.
.
}
\{\{s_1,s_2,s_3,s_4\}_1,\{s_1,s_2,s_3,s_4\}_2,...\}
{{s1,s2,s3,s4}1,{s1,s2,s3,s4}2,...} ,表示集合中每一种状态
{
s
1
,
s
2
,
s
3
,
s
4
}
i
\{s_1,s_2,s_3,s_4\}_i
{s1,s2,s3,s4}i 都可行。不同的
{
s
1
,
s
2
,
s
3
,
s
4
}
\{s_1,s_2,s_3,s_4\}
{s1,s2,s3,s4} 一共有
2
4
=
16
2^4=16
24=16 种,不妨就记为二进制数
0000—1111
\text{0000---1111}
0000—1111 ,那我们就可以用
2
16
2^{16}
216 以内的二进制数来表示状态集,第
i
i
i 位上表示
0000—1111
\text{0000---1111}
0000—1111 这 16 种状态的第
i
i
i 种是否可能。(状态有
2
4
=
16
2^4=16
24=16 种,状态集有
2
2
4
=
2
16
2^{2^4}=2^{16}
224=216 种)
于是我们可以设计一个状态数为
log
T
∗
2
16
\log T*2^{16}
logT∗216 的 DP:
d
p
[
i
]
[
S
]
dp[i][S]
dp[i][S] 表示进行到第
i
i
i 位,状态集为
S
S
S. 转移的时候就把状态集
S
S
S 中每一种可能的状态进行转移,然后处理出下一位每一种状态是否可行,以便处理出新的状态集
S
0
′
S'_0
S0′ 和
S
1
′
S'_1
S1′(对应当前位的与值为 0 或 1)。
这应该是做过的单次复杂度最高的数位 DP 了,复杂度为
O
(
log
T
∗
2
16
∗
16
)
O(\log T*2^{16}*16)
O(logT∗216∗16) (按道理复杂度中不应该带常数的)。
CODE
#include<set>
#include<map>
#include<cmath>
#include<queue>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
#define MAXN 100005
#define DB double
#define LL long long
#define ENDL putchar('\n')
#define lowbit(x) (-(x) & (x))
#pragma GCC optimize(2)
LL read() {
LL f = 1,x = 0;char s = getchar();
while(s < '0' || s > '9') {if(s=='-')f = -f;s = getchar();}
while(s >= '0' && s <= '9') {x=x*10+(s-'0');s = getchar();}
return f * x;
}
const int MOD = 998244353;
int n,m,i,j,s,o,k;
LL T,l1,r1,l2,r2;
LL dp[65][1<<16|5];
int cg(int i,int ss,int x,int y) {
int s1 = (ss&1),s2 = ((ss>>1)&1),s3 = ((ss>>2)&1),s4 = ((ss>>3)&1);
int b1 = (s1 ? ((l1>>i)&1):0),t1 = (s2 ? (((r1&T)>>i)&1):((T>>i)&1));
int b2 = (s3 ? ((l2>>i)&1):0),t2 = (s4 ? (((r2&T)>>i)&1):((T>>i)&1));
if(x < b1 || x > t1 || y < b2 || y > t2 || (x|y) != ((T>>i)&1)) return 16;
int res = 0;
if(x == ((l1>>i)&1) && s1) res |= 1;
if(x == ((r1>>i)&1) && s2) res |= (1<<1);
if(y == ((l2>>i)&1) && s3) res |= (1<<2);
if(y == ((r2>>i)&1) && s4) res |= (1<<3);
return res;
}
int main() {
freopen("number.in","r",stdin);
freopen("number.out","w",stdout);
T = read();l1 = read();r1 = read();l2 = read();r2 = read();
dp[60][1<<15] = 1;
for(int i = 59;i >= 0;i --) {
for(int S = 0;S < (1<<16);S ++) {
if(!dp[i+1][S]) continue;
int S1 = 0,S0 = 0;
for(int j = 0;j < 16;j ++) {
if(!(S & (1<<j))) continue;
S1 |= (1<<cg(i,j,1,1));
S0 |= (1<<cg(i,j,1,0));
S0 |= (1<<cg(i,j,0,1));
S0 |= (1<<cg(i,j,0,0));
}
S1 &= ((1<<16)-1);
S0 &= ((1<<16)-1);
if(S1) dp[i][S1] += dp[i+1][S];
if(S0) dp[i][S0] += dp[i+1][S];
}
}
LL ans = 0;
for(int i = 1;i < (1<<16);i ++) {
ans += dp[0][i];
}
printf("%lld\n",ans);
return 0;
}
造数字(数位DP)的更多相关文章
- xbz分组题B 吉利数字 数位dp入门
B吉利数字时限:1s [题目描述]算卦大湿biboyouyun最近得出一个神奇的结论,如果一个数字,它的各个数位相加能够被10整除,则称它为吉利数.现在叫你计算某个区间内有多少个吉利数字. [输入]第 ...
- 牛客小白月赛8 - E - 诡异数字 数位DP
牛客小白月赛8 - E - 诡异数字 题意: 求区间中,满足限制条件的数字的个数. 限制条件就是某些数字不能连续出现几次. 思路: 比较裸的数位DP, DP数组开一个dp[len][x][cnt] 表 ...
- SCUT - 289 - 小O的数字 - 数位dp
https://scut.online/p/289 一个水到飞起的模板数位dp. #include<bits/stdc++.h> using namespace std; typedef ...
- HDU 4352 XHXJ's LIS HDU(数位DP)
HDU 4352 XHXJ's LIS HDU 题目大意 给你L到R区间,和一个数字K,然后让你求L到R区间之内满足最长上升子序列长度为K的数字有多少个 solution 简洁明了的题意总是让人无从下 ...
- 【BZOJ-1833】count数字计数 数位DP
1833: [ZJOI2010]count 数字计数 Time Limit: 3 Sec Memory Limit: 64 MBSubmit: 2494 Solved: 1101[Submit][ ...
- BZOJ_1833_[ZJOI2010]_数字计数_(数位dp)
描述 http://www.lydsy.com/JudgeOnline/problem.php?id=1833 统计\(a~b\)中数字\(0,1,2,...,9\)分别出现了多少次. 分析 数位dp ...
- zznu 1255 数字统计(数位DP, 数学方法)
最近在学数位DP, 感觉还是满有收获的! 做了几个题之后想起来自己OJ上曾经做的一道题,以前是用数学方法写的,现在改用数位DP来写了一遍. 题目: 1255: 数字统计 时间限制: 1 Sec 内存 ...
- BZOJ_1833_[ZJOI2010]count 数字计数_数位DP
BZOJ_1833_[ZJOI2010]count 数字计数_数位DP 题意: 给定两个正整数a和b,求在[a,b]中的所有整数中,每个数码(digit)各出现了多少次. 分析: 数位DP f[i][ ...
- BZOJ 1833 数字计数 数位DP
题目链接 做的第一道数位DP题,听说是最基础的模板题,但还是花了好长时间才写出来..... 想深入了解下数位DP的请点这里 先设dp数组dp[i][j][k]表示数位是i,以j开头的数k出现的次数 有 ...
随机推荐
- idea中一些常用的快捷键
ctrl+shift+alt:多行操作psvm:生成main()方法:fori:生成for循环:Ctrl+Alt+v:自动补齐返回值类型ctrl+o:覆写方法ctrl+i:实现接口中的方法ctrl+s ...
- DAST 黑盒漏洞扫描器 第六篇:运营篇(终)
0X01 前言 转载请标明来源:https://www.cnblogs.com/huim/ 当项目功能逐渐成熟,同时需要实现的是运营流程和指标体系建设.需要工程化的功能逐渐少了,剩下的主要工作转变成持 ...
- UiPath键盘操作的介绍和使用
一.键盘操作的介绍 模拟用户使用键盘操作的一种行为: 例如使用发送热键(Sendhotkey),输入信息 (Typeinto)的操作 二.键盘操作在UiPath中的使用 1.打开设计器,在设计库中新建 ...
- 多校联训 DS 专题
CF1039D You Are Given a Tree 容易发现,当 \(k\) 不断增大时,答案不断减小,且 \(k\) 的答案不超过 \(\lfloor\frac {n}{k}\rfloor\) ...
- 强化学习-学习笔记7 | Sarsa算法原理与推导
Sarsa算法 是 TD算法的一种,之前没有严谨推导过 TD 算法,这一篇就来从数学的角度推导一下 Sarsa 算法.注意,这部分属于 TD算法的延申. 7. Sarsa算法 7.1 推导 TD ta ...
- Unity3D学习笔记8——GPU实例化(3)
目录 1. 概述 2. 详论 2.1. 自动实例化 2.2. MaterialPropertyBlock 3. 参考 1. 概述 在前两篇文章<Unity3D学习笔记6--GPU实例化(1)&g ...
- 树莓派实战:微信机器人(itchat实现)
背景 楼主有一台树莓派4B开发板(8G内存版),是目前的顶配机型.这一年来的业余时间,除了写Java.架构方面的文章,也陆续折腾了不少树莓派上的好玩小项目,在此新开一个树莓派实战的文章系列,分享给粉丝 ...
- centos7 yum error yum doesn't have enough cached data
1.vi /etc/resolv.conf,添加下面一行 nameserver 114.114.114.114 修改完成后service network restart进行重启,试一下yum upda ...
- 编译安装Python出现“pip is configured with locations that require TLS/SSL, however the ssl.....”
ubuntu: sudo apt-get install libssl-dev Cenos: sudo yum install openssl-devel 重新编译: ./configure --en ...
- 绿色安装MySQL5.7版本----配置my.ini文件注意事项
前言 由于前段时间电脑重装,虽然很多软件不在C盘,但是由于很多注册表以及关联文件被删除,很多软件还需要重新配置甚至卸载重装. 使用MySQL时就遇到了这种情况,在修改配置文件无效的情况下选择了重新安装 ...