Problem Description
As we know, Rikka is poor at math. Yuta is worrying about this situation, so he gives Rikka some math tasks to practice. There is one of them:

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?
 

Input
The first line contains a number T(T≤10)——The
number of the testcases.

For each testcase, the first line contains a number n(n≤10300).
 

Output
For each testcase, print a single number. The answer may be very large, so you only need to print the answer modulo 998244353.
 

Sample Input

1
10
 

Sample Output

7

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+十进制转化为二进制)的更多相关文章

  1. 十进制转化为二进制Java实现

    提取2的幂 这个方法用代码实现貌似有点麻烦,需要探测大小,我只实现了整数十进制到二进制的转化 /* * 提取2的幂 */ public static String TenToBin1(int ten) ...

  2. 用JS实现将十进制转化为二进制

  3. 数位dp从会打模板到不会打模板

    打了几个数位$dp$,发现自己除了会打模板之外没有任何长进,遇到非模板题依然什么都不会 那么接下来这篇文章将介绍如何打模板(滑稽) 假设我们要处理$l----r$ 采用记忆化搜索的方式,枚举$< ...

  4. ACM学习历程—HDU5587 Array(数学 && 二分 && 记忆化 || 数位DP)(BestCoder Round #64 (div.2) 1003)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5587 题目大意就是初始有一个1,然后每次操作都是先在序列后面添加一个0,然后把原序列添加到0后面,然后 ...

  5. 浅谈数位DP

    在了解数位dp之前,先来看一个问题: 例1.求a~b中不包含49的数的个数. 0 < a.b < 2*10^9 注意到n的数据范围非常大,暴力求解是不可能的,考虑dp,如果直接记录下数字, ...

  6. 初探数位dp

    数位dp有着很明显的特点,一般来说是给定区间[l,r]求满足某种条件区间中的数有多少个 朴素解法一般是O(n)的而n往往很大(10^8起步) 这时候我们就要想办法优化,于是就有了数位dp 数位有两个基 ...

  7. 专题训练之数位DP

    推荐以下一篇博客:https://blog.csdn.net/wust_zzwh/article/details/52100392 1.(HDOJ2089)http://acm.hdu.edu.cn/ ...

  8. Codeforces - 914C 数位DP

    题意有点难以描述,简略的就是给定一个二进制\(n\),每一步操作能使\(n\)的位为1的数的和转化为一个十进制,然后转化为该数的二进制再进行相同的操作 查询\([0,n]\)中操作数恰好为\(k\)的 ...

  9. 开坑数位dp

    [背景] 在10月3日的dp专练中,压轴的第6题是一道数位dp,于是各种懵逼. 为了填上这个留存已久的坑,蒟蒻chty只能开坑数位dp了. [例题一][HDU2089]不要62 题目大意:给你一个区间 ...

随机推荐

  1. 美业黑科技 ▏肌肤管家SkinRun V3S智能肌肤测试仪,实现“护肤”私人定制

    肌肤如同身体,也需要定时的"健康检查",但仅凭肉眼难以窥见深层的肌肤问题.而现在,肌肤管家SkinRun前沿黑科技护肤测试仪--SkinRun V3S便能帮助用户对症下药.肌肤管家 ...

  2. java环境配置-win10(傻瓜式教程)

    java环境配置 – 小学弟要开始学java了,由于本人较懒,表达能力有限,所以来这,写一篇简单的指导,帮学弟装下java环境. 首先打开浏览器,输入这个网址https://www.oracle.co ...

  3. 【Spring Boot】创建一个简单的Spring Boot的 Demo

    走进Spring Boot 文章目录 走进Spring Boot 环境搭建 新建Spring Boot项目 开始创建项目 配置JDK版本 和 Initializr Service URL 配置Proj ...

  4. 1.搭建Hadoop实验平台

    节点功能规划 操作系统:CentOS7.2(1511) Java JDK版本:jdk-8u65-linux-x64.tar.gz Hadoop版本:hadoop-2.8.3.tar.gz 下载地址: ...

  5. qt for webassembly环境搭建图文教程

    一.前言 从Qt5.14开始,官方的在线安装提供了qt for webassembly构建套件,这对很多小白来说绝对是个好消息,也绝对是个好东西,好消息是不用再去交叉编译自己生成qt for weba ...

  6. Java运算符概要与数学函数

    运算符概要 在Java中,使用算术运算符+,-,*,/表示加减乘除运算,当参与/的运算的两个操作数都是整数时,表示整数除法,否则,表示浮点除法.整数的求余操作(有时称为取模),用%表示,例如,15/2 ...

  7. centos7搭建dolphinscheduler集群

    一.简述 Apache DolphinScheduler是一个分布式去中心化,易扩展的可视化DAG工作流任务调度系统.致力于解决数据处理流程中错综复杂的依赖关系,使调度系统在数据处理流程中开箱即用.有 ...

  8. hive搜索报错

    在自己搭建的集群上执行hive搜索语句 select count(*) from ods_event_log where dt='2019-12-14' group by dt; 报错如下: Stat ...

  9. 为失败设计,大量引入对SRE的理解,鲁棒性高

    https://go-kratos.dev/#/ Principles 简单:不过度设计,代码平实简单 通用:通用业务开发所需要的基础库的功能 高效:提高业务迭代的效率 稳定:基础库可测试性高,覆盖率 ...

  10. (Mysql)连接问题之1130

    访问远程服务器上的Mysql数据库连接是报:1130-host is not allowed to connect this MYSQL severe; 解决方案: 登录远程服务器下的mysql下. ...