先复习了下之前做的数位DP又做了道新题才看的这道题,对我来说还是一种新类型,涉及到非线性计算,之前做的都是形如 \(dp[x]-dp[y]\)这样的只用处理一个上限做下差即可。一开始想分别枚举 \(x\)和 \(y\)中最高位 \(1\)的位置,计算符合要求的组合数目,但是一旦两个数的最高位 \(1\)是上限的时候感觉处理起来会很麻烦,需要分很多种讨论,而且情况还不好处理,容易缺漏。然后看了下题解,它是数位DP的思想加上记忆化搜索,它直接枚举最高位 \(1\)的位置,然后计算有多少种组合满足要求。状态数组为 \(dp[bit][is\_1][is\_2][ok]\),第一维代表枚举的位数,第二、三维代表两个数的第 \(bit\)位是否被锁定,如果被锁定则该位取值不能大于原数该位的值,实际上是在限制这两个数不能大于原数,第四维代表两数的或运算结果在第 \(bit\)位是否必须为 \(1\)。然后记忆化搜索统计答案。

#include<cstdio>
#include<cstring>
const int mod = 1e9 + 7; int T, x, y;
int dp[35][2][2][2]; inline int max(int a, int b) { return a > b ? a : b; } // ok=1代表当前枚举的两个数的第bit位的|必须为1
int dfs(int bit, bool is_1, bool is_2, bool ok){
if(bit == -1) return 1;
if(dp[bit][is_1][is_2][ok]) return dp[bit][is_1][is_2][ok];
int temp = 0;
// 上界
int k1 = is_1 ? (x >> bit & 1) : 1;
int k2 = is_2 ? (y >> bit & 1) : 1;
//printf("%d %d %d", bit, k1, k2);
for(int i = 0; i <= k1; ++i)
for(int t = 0; t <= k2; ++t){
if(i & t) continue;
// 此后不必要求两数之|为1,ok=0
if(ok){
if(i | t) temp = (temp + dfs(bit - 1, is_1 && (i == k1), is_2 && (t == k2), 0)) % mod;
}
else{
temp = (temp + dfs(bit - 1, is_1 && (i == k1), is_2 && (t == k2), 0)) % mod;
}
}
return dp[bit][is_1][is_2][ok] = temp;
} int main(){
scanf("%d", &T);
while(T--){
int ans = 0;
memset(dp, 0, sizeof(dp));
scanf("%d%d", &x, &y);
int a = x, b = y, da = -1, db = -1;
for(; a; a >>= 1) ++da;
for(; b; b >>= 1) ++db;
for(int i = max(da, db); i >= 0; --i){
// 因为正在统计第i位的答案,只有两数之|在第i位为1才有贡献,所以ok=1
ans = (ans + 1LL * dfs(i, i >= da, i >= db, 1) * (i + 1)) % mod;
}
printf("%d\n", ans);
}
return 0;
}

2020ICPC上海 C题(数位dp, 记忆化搜索)的更多相关文章

  1. [hihocoder 1033]交错和 数位dp/记忆化搜索

    #1033 : 交错和 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描写叙述 给定一个数 x,设它十进制展从高位到低位上的数位依次是 a0, a1, ..., an - 1 ...

  2. 【poj1850】 Code 数位dp+记忆化搜索

    题目大意:给你一个字符串,问你这个字符串的rank,如果这个字符串不合法,请直接输出0.(一个合法的字符串是对于∀i,有c[i]<c[i+1]) 字符串s的rank的计算方式:以字符串长度作为第 ...

  3. bzoj1833: [ZJOI2010]count 数字计数(数位DP+记忆化搜索)

    1833: [ZJOI2010]count 数字计数 题目:传送门 题解: 今天是躲不开各种恶心DP了??? %爆靖大佬啊!!! 据说是数位DP裸题...emmm学吧学吧 感觉记忆化搜索特别强: 定义 ...

  4. 1026-windy数+数位DP+记忆化搜索

    1026: [SCOI2009]windy数 题意:数位DP模板题: 目前只理解了记忆化搜索,就想练练手, ------给递推写法留一个位子 ------ 注意这道题要判断前导0的情况,1 )可以加一 ...

  5. 数位dp/记忆化搜索

    一.引例 #1033 : 交错和 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 给定一个数 x,设它十进制展从高位到低位上的数位依次是 a0, a1, ..., an  ...

  6. [BZOJ3598][SCOI2014]方伯伯的商场之旅(数位DP,记忆化搜索)

    3598: [Scoi2014]方伯伯的商场之旅 Time Limit: 30 Sec  Memory Limit: 64 MBSubmit: 449  Solved: 254[Submit][Sta ...

  7. 【每日dp】 Gym - 101889E Enigma 数位dp 记忆化搜索

    题意:给你一个长度为1000的串以及一个数n 让你将串中的‘?’填上数字 使得该串是n的倍数而且最小(没有前导零) 题解:dp,令dp[len][mod]为是否出现过 填到第len位,余数为mod 的 ...

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

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

  9. hdu3652 数位dp记忆化搜索

    从未见过的船新版本数位dp,,省去了预处理过程,直接进行计算 #include<bits/stdc++.h> using namespace std; #define ll long lo ...

  10. cf55D 数位dp记忆化搜索+状态离散

    /* 漂亮数定义:可以整除任意数位上的数 求出区间[l,r]之间的漂亮数个数 因为 dp[i][j][k]:i位前模lcm的值是j,i位前lcm是k的漂亮数个数 */ #include<bits ...

随机推荐

  1. [golang]gin框架接收websocket通信

    前言 WebSocket是一种在单个TCP连接上进行全双工通信的协议.WebSocket让客户端和服务端之间的数据交换变得非常简单,且允许服务器主动向客户端推送数据,并且之后客户端和服务端所有的通信都 ...

  2. [linux]搭建nfs

    环境 说明 IP 系统版本 服务端 192.168.137.7 centos 7 客户端 192.168.137.8 centos 7 步骤 配置服务端 安装 nfs 服务 yum install - ...

  3. debian11安装mysql5.7

    前言 mysql官网5.7版本的只找到debian10的,没有debian11的,试了下也能用. 系统版本:debian 11 mysql版本:5.7.35 步骤 下载bundle的tar包.官网地址 ...

  4. Android OAID 获取 基于MSA oaid_sdk_1.0.25.zip

    目录 简介 SDK下载 支持版本 开始使用 可能遇到的问题 替代方案 APK下载 写在最后 简介 本文链接:https://www.cnblogs.com/dks-/p/14357041.html 根 ...

  5. 【实践篇】DDD脚手架及编码规范

    一.背景介绍 我们团队一直在持续推进业务系统的体系化治理工作,在这个过程中我们沉淀了自己的DDD脚手架项目.脚手架项目是体系化治理过程中比较重要的一环,它的作用有两点: (1)可以对新建的项目进行统一 ...

  6. TIDB - 分布式数据库

    TIDB(一) 重点 TIDB核心 数据存储-RocksDB Raft 协议 选举 数据同步 MVCC 表数据与kv映射关系 索引数据与kv 映射关系 元数据和sql 层计算 PD调度 HTAP 特性 ...

  7. 通过Scrum实现最大生产力的五种方法

    在数字化.信息化.智能化蓬勃发展的今天,敏捷开发和Scrum已成为重塑项目管理的重要方式. 敏捷是一种体现不同方法的思维方式,包括了Scrum,看板,极限编程(XP).精益开发等众多框架. Scrum ...

  8. 使用Skonsole自动生成Git提交信息

    使用Skonsole自动生成Git提交信息 随着LLM应用的普及,日常工作中的很多使用都可以使用LLM来完成,比如Git提交信息的生成. Skonsole是一个基于Semantic Kernel的命令 ...

  9. Hadoop核心概念

    大数据开发总体架构: Hadoop是大数据开发所使用的一个核心框架.使用Hadoop可以方便的管理分布式集群,将海量数据分布式的存储在集群中,并使用分布式并行程序来处理这些数据. Hadoop由许多子 ...

  10. git + docker + docker-compose + Jenkins+Linux 自动化构建、部署、测试过程

    Jenkins三大概念: Job:即为任务 插件:maven构建项目.git拉取代码.ssh插件 工作空间:任务的存储空间,即为git代码的存储空间 开发者在本地开发,然后提交到 Source Res ...