hdu5432Rikka with Array (数位dp+十进制转化为二进制)
Yuta has an array A of
length n,and
the ith
element of A is
equal to the sum of all digits of i in
binary representation. For example,A[1]=1,A[3]=2,A[10]=2.
Now, Yuta wants to know the number of the pairs (i,j)(1≤i<j≤n) which
satisfy A[i]>A[j].
It is too difficult for Rikka. Can you help her?
number of the testcases.
For each testcase, the first line contains a number n(n≤10300).
10
When $n=10$, $A$ is equal to $1,1,2,1,2,2,3,1,2,2$.
So the answer is $7$.
题意:定义A[i]为i化为二进制后的1的个数,给你一个数n,你要在1~n之间找到两个数a,b,使得a<b且A[a]>A[b].问总共有多少这样的对数.
思路:这题和普通的数位dp不同,普通的数位dp是求单个数内的方案数,但是这道题是求对数,所以不能用之前的方法。这道题中我们要在记忆化搜索中同时枚举两个数,一个为大数,一个为小数,然后用dp[pos][cha][flag][same][lim]表示前pos位,大小两数化为二进制后1的个数差的绝对值为cha,flag表示是否小数化为二进制后的1的个数大于大数,same表示pos位前两个数是不是完全相同,lim表示大数是不是还有限制。据说题解只开了三维,感觉三维的条件太少了,我只会写五维的。。
#include<iostream>
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<math.h>
#include<vector>
#include<map>
#include<set>
#include<queue>
#include<stack>
#include<string>
#include<bitset>
#include<algorithm>
using namespace std;
typedef long long ll;
typedef long double ldb;
#define inf 99999999
#define pi acos(-1.0)
#define maxn 1005
#define MOD 998244353
char s[maxn];
int dp[maxn][maxn][2][2][2]; //dp[pos][cha][flag1][same][lim]表示前pos位,大小数1的个数差的绝对值是cha,小数中1的个数是不是大于大数中的个数
int wei[maxn],num[maxn];
int dfs(int pos,int cha,int flag,int same,int lim) //same表示大数和小数是不是前几位都一样
{
int i,j;
if(pos==0){
if(flag==1)return 1;
return 0;
}
if(dp[pos][cha][flag][same][lim]!=-1 ){
return dp[pos][cha][flag][same][lim];
}
int ans=0;
if(same==1){
if(lim==1){
if(wei[pos]==0){
ans=ans+dfs(pos-1,0,0,1,1); //都取0
if(ans>=MOD)ans-=MOD;
}
else if(wei[pos]==1){
ans=ans+dfs(pos-1,0,0,1,1);if(ans>=MOD)ans-=MOD; //都取1
ans=ans+dfs(pos-1,0,0,1,0);if(ans>=MOD)ans-=MOD; //都取0
ans=ans+dfs(pos-1,1,0,0,1);if(ans>=MOD)ans-=MOD; //大数取1,小数取0
}
}
else if(lim==0){
ans=ans+dfs(pos-1,0,0,1,0);if(ans>=MOD)ans-=MOD;
ans=ans+dfs(pos-1,0,0,1,0);if(ans>=MOD)ans-=MOD;
ans=ans+dfs(pos-1,1,0,0,0);if(ans>=MOD)ans-=MOD;
}
}
else if(same==0){
if(lim==1){
if(wei[pos]==1){
ans=ans+dfs(pos-1,cha,flag,same,0);if(ans>=MOD)ans-=MOD; //都加0
ans=ans+dfs(pos-1,cha,flag,same,1);if(ans>=MOD)ans-=MOD; //都加1
if(cha==0){
ans=ans+dfs(pos-1,1,0,same,1);if(ans>=MOD)ans-=MOD; //大数加1
ans=ans+dfs(pos-1,1,1,same,0);if(ans>=MOD)ans-=MOD; //小数加1
}
else if(cha>0){
if(flag==1){
int flag1;
if(cha==1)flag1=0;
else flag1=1;
ans=ans+dfs(pos-1,cha-1,flag1,same,1);if(ans>=MOD)ans-=MOD; //大数加1
ans=ans+dfs(pos-1,cha+1,1,same,0);if(ans>=MOD)ans-=MOD; //小数加1
}
else if(flag==0){
ans=ans+dfs(pos-1,cha+1,0,same,1 );if(ans>=MOD)ans-=MOD; //大数加1
ans=ans+dfs(pos-1,cha-1,0,same,0);if(ans>=MOD)ans-=MOD; //小数加1
}
}
}
else if(wei[pos]==0){
ans=ans+dfs(pos-1,cha,flag,same,1);if(ans>=MOD)ans-=MOD; //都加0
if(cha==0){
ans=ans+dfs(pos-1,cha+1,1,same,1);if(ans>=MOD)ans-=MOD; //大数加0,小数加1
}
else if(cha>0){
if(flag==1){
ans=ans+dfs(pos-1,cha+1,1,same,1);if(ans>=MOD)ans-=MOD;
}
else if(flag==0){
ans=ans+dfs(pos-1,cha-1,0,same,1);if(ans>=MOD)ans-=MOD;
}
}
}
}
else if(lim==0){
ans=ans+dfs(pos-1,cha,flag,same,0);if(ans>=MOD)ans-=MOD; //都加0
ans=ans+dfs(pos-1,cha,flag,same,0);if(ans>=MOD)ans-=MOD; //都加1
if(cha==0){
ans=ans+dfs(pos-1,1,0,same,0);if(ans>=MOD)ans-=MOD; //大数加1
ans=ans+dfs(pos-1,1,1,same,0);if(ans>=MOD)ans-=MOD; //小数加1
}
else if(cha>0){
if(flag==1){
int flag1;
if(cha==1)flag1=0;
else flag1=1;
ans=ans+dfs(pos-1,cha-1,flag1,same,0 );if(ans>=MOD)ans-=MOD; //大数加1
ans=ans+dfs(pos-1,cha+1,1,same,0);if(ans>=MOD)ans-=MOD; //小数加1
}
else if(flag==0){
ans=ans+dfs(pos-1,cha+1,0,same,0 );if(ans>=MOD)ans-=MOD; //大数加1
ans=ans+dfs(pos-1,cha-1,0,same,0);if(ans>=MOD)ans-=MOD; //小数加1
}
}
}
}
dp[pos][cha][flag][same][lim]=ans;
return ans;
}
void solve()
{
int i,j;
int len=strlen(s+1);
for(i=len;i>=1;i--){
num[i]=s[len+1-i]-'0';
}
int tot=0;
while(len){
for(i=len-1;i>=1;i--){
num[i]+=(num[i+1]&1)*10;
num[i+1]>>=1;
}
if(num[1]&1)wei[++tot]=1;
else wei[++tot]=0;
num[1]>>=1;
if(num[len]==0)len--;
}
memset(dp,-1,sizeof(dp)); //这里要注意,必须每一个样例都要初始化一遍,因为不同的数,dp[pos][cha][flag1][same][lim]的意义不同
printf("%d\n",dfs(tot,0,0,1,1)%MOD);
}
int main()
{
int n,m,i,j,T;
scanf("%d",&T);
while(T--)
{
scanf("%s",s+1);
solve();
}
return 0;
}
hdu5432Rikka with Array (数位dp+十进制转化为二进制)的更多相关文章
- 十进制转化为二进制Java实现
提取2的幂 这个方法用代码实现貌似有点麻烦,需要探测大小,我只实现了整数十进制到二进制的转化 /* * 提取2的幂 */ public static String TenToBin1(int ten) ...
- 用JS实现将十进制转化为二进制
- 数位dp从会打模板到不会打模板
打了几个数位$dp$,发现自己除了会打模板之外没有任何长进,遇到非模板题依然什么都不会 那么接下来这篇文章将介绍如何打模板(滑稽) 假设我们要处理$l----r$ 采用记忆化搜索的方式,枚举$< ...
- ACM学习历程—HDU5587 Array(数学 && 二分 && 记忆化 || 数位DP)(BestCoder Round #64 (div.2) 1003)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5587 题目大意就是初始有一个1,然后每次操作都是先在序列后面添加一个0,然后把原序列添加到0后面,然后 ...
- 浅谈数位DP
在了解数位dp之前,先来看一个问题: 例1.求a~b中不包含49的数的个数. 0 < a.b < 2*10^9 注意到n的数据范围非常大,暴力求解是不可能的,考虑dp,如果直接记录下数字, ...
- 初探数位dp
数位dp有着很明显的特点,一般来说是给定区间[l,r]求满足某种条件区间中的数有多少个 朴素解法一般是O(n)的而n往往很大(10^8起步) 这时候我们就要想办法优化,于是就有了数位dp 数位有两个基 ...
- 专题训练之数位DP
推荐以下一篇博客:https://blog.csdn.net/wust_zzwh/article/details/52100392 1.(HDOJ2089)http://acm.hdu.edu.cn/ ...
- Codeforces - 914C 数位DP
题意有点难以描述,简略的就是给定一个二进制\(n\),每一步操作能使\(n\)的位为1的数的和转化为一个十进制,然后转化为该数的二进制再进行相同的操作 查询\([0,n]\)中操作数恰好为\(k\)的 ...
- 开坑数位dp
[背景] 在10月3日的dp专练中,压轴的第6题是一道数位dp,于是各种懵逼. 为了填上这个留存已久的坑,蒟蒻chty只能开坑数位dp了. [例题一][HDU2089]不要62 题目大意:给你一个区间 ...
随机推荐
- 美业黑科技 ▏肌肤管家SkinRun V3S智能肌肤测试仪,实现“护肤”私人定制
肌肤如同身体,也需要定时的"健康检查",但仅凭肉眼难以窥见深层的肌肤问题.而现在,肌肤管家SkinRun前沿黑科技护肤测试仪--SkinRun V3S便能帮助用户对症下药.肌肤管家 ...
- java环境配置-win10(傻瓜式教程)
java环境配置 – 小学弟要开始学java了,由于本人较懒,表达能力有限,所以来这,写一篇简单的指导,帮学弟装下java环境. 首先打开浏览器,输入这个网址https://www.oracle.co ...
- 【Spring Boot】创建一个简单的Spring Boot的 Demo
走进Spring Boot 文章目录 走进Spring Boot 环境搭建 新建Spring Boot项目 开始创建项目 配置JDK版本 和 Initializr Service URL 配置Proj ...
- 1.搭建Hadoop实验平台
节点功能规划 操作系统:CentOS7.2(1511) Java JDK版本:jdk-8u65-linux-x64.tar.gz Hadoop版本:hadoop-2.8.3.tar.gz 下载地址: ...
- qt for webassembly环境搭建图文教程
一.前言 从Qt5.14开始,官方的在线安装提供了qt for webassembly构建套件,这对很多小白来说绝对是个好消息,也绝对是个好东西,好消息是不用再去交叉编译自己生成qt for weba ...
- Java运算符概要与数学函数
运算符概要 在Java中,使用算术运算符+,-,*,/表示加减乘除运算,当参与/的运算的两个操作数都是整数时,表示整数除法,否则,表示浮点除法.整数的求余操作(有时称为取模),用%表示,例如,15/2 ...
- centos7搭建dolphinscheduler集群
一.简述 Apache DolphinScheduler是一个分布式去中心化,易扩展的可视化DAG工作流任务调度系统.致力于解决数据处理流程中错综复杂的依赖关系,使调度系统在数据处理流程中开箱即用.有 ...
- hive搜索报错
在自己搭建的集群上执行hive搜索语句 select count(*) from ods_event_log where dt='2019-12-14' group by dt; 报错如下: Stat ...
- 为失败设计,大量引入对SRE的理解,鲁棒性高
https://go-kratos.dev/#/ Principles 简单:不过度设计,代码平实简单 通用:通用业务开发所需要的基础库的功能 高效:提高业务迭代的效率 稳定:基础库可测试性高,覆盖率 ...
- (Mysql)连接问题之1130
访问远程服务器上的Mysql数据库连接是报:1130-host is not allowed to connect this MYSQL severe; 解决方案: 登录远程服务器下的mysql下. ...