神仙题。为啥我第一眼看上去以为是个普及题

路径有两种,第一种是从 LCA 一边下去的,第二种是从 LCA 两边都下去了的。

先考虑第一种。

先枚举路径长度 \(h\)。

当 LCA 编号是 \(x\) 时,且所有儿子都是往左走时,和为 \((2^h-1)x\);所有儿子都往右走时,和为 \((2^h-1)x+2^h-1\)。显然 \((2^h-1)x\le s\le (2^h-1)x+2^h-1\)。

考虑从下往上第 \(i\) 个点从左儿子变成右儿子时(其它不变),总和会增加 \(2^i-1\)。

接下来我们发现无论这条路径长什么样,\(x\) 都等于 \(\lfloor\frac{s}{2^h-1}\rfloor\)。因为当 \(x\) 更小时,即使所有儿子都是往右走的,也没有 \(x\) 取 \(\lfloor\frac{s}{2^h-1}\rfloor\) 时所有儿子都往左走的和大,自然不可能和为 \(s\)。\(x\) 更大时同理。

所以我们想让总和再增加 \(s-(2^h-1)x\)。问题就是有多少种方案,从 \(2^1-1,2^2-1,\dots,2^{h-1}-1\) 中选出一些数(LCA 不能选),使得和为 \(s-(2^h-1)x\)。

这就简单了。枚举选了 \(cnt\) 个数,就是能否从 \(2^1,2^2,\dots,2^{h-1}\) 中选出一些数使得和为 \(s-(2^h-1)x+cnt\)。当且仅当 \(s-(2^h-1)x+cnt\) 中 \(1\) 的个数恰好为 \(cnt\) 方案为 \(1\),否则为 \(0\)。

接下来考虑第二种。

同样的,枚举左链和右链的长度 \(l\), \(r\)(都包括 LCA,至少我是这么写的)。同理可以推出 \(x\) 恒等于 \(\lfloor\frac{s-2^{r-1}+1}{2^l+2^r-3}\rfloor\)。

同理,考虑从全部是左儿子变成一些右儿子。(当然,LCA 的两个儿子除外)

问题就是有多少种方案,从 \(2^1-1,2^2-1,\dots,2^{l-1}-1,2^1-1,2^2-1,\dots,2^{r-1}-1\) 中选出一些数,使得和为 \(s-2^{r-1}+s-(2^l+2^r-3)x\)。

同样枚举个数 \(cnt\)。下文为了方便设 \(res=s-2^{r-1}+s-(2^l+2^r-3)x+cnt\)。

这回没办法了,老实上 DP。

令 \(f[i][j][k]\) 表示考虑 \(2^1\) 到 \(2^i\) 这些数,从中选出了 \(j\) 个,上一位有没有向这一位进位(\(k\) 是 01 变量)。

初始状态有 \(f[0][0][0]=1\)。要求是 \(f[\max(l,r)][cnt][0]\)。(由于选完之后不能再进位,所以 \(k=0\),此时 \(i=\max(l,r)\) 会更方便)

转移方程,枚举左子树选不选(设为 \(a\)),右子树选不选(设为 \(b\)),\(f[i+1][j+a+b][\lfloor\frac{k+a+b}{2}\rfloor]+=f[i][j][k]\)。

转移条件,首先对应的数要能选(即 \(i+1\ge l-1\) 时就选不了左子树了),另外 \(res\) 的第 \(i+1\) 位应恰好是 \((k+a+b)\bmod 2\)。

时间复杂度 \(O(\log^5s)\)。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=100010,mod=998244353;
#define lson o<<1,l,mid
#define rson o<<1|1,mid+1,r
#define FOR(i,a,b) for(int i=(a);i<=(b);i++)
#define ROF(i,a,b) for(int i=(a);i>=(b);i--)
#define MEM(x,v) memset(x,v,sizeof(x))
inline int read(){
int x=0,f=0;char ch=getchar();
while(ch<'0' || ch>'9') f|=ch=='-',ch=getchar();
while(ch>='0' && ch<='9') x=x*10+ch-'0',ch=getchar();
return f?-x:x;
}
ll n,f[55][111][2];
int bitcnt(ll x){
int c=0;
for(;x;x&=x-1) c++;
return c;
}
ll solve1(){
ll ans=0;
FOR(i,1,50){
ll x=n/((1ll<<i)-1);
if(!x) break;
ll res=n-x*((1ll<<i)-1);
FOR(j,0,i-1) if((res+j)%2==0 && bitcnt(res+j)==j) ans++;
}
return ans;
}
ll solve2(){
ll ans=0;
FOR(l,2,50) FOR(r,2,50){
ll x=(n-(1ll<<(r-1))+1)/((1ll<<l)+(1ll<<r)-3);
if(!x) break;
// printf("x=%lld\n",x);
ll res=n-(1ll<<(r-1))+1-x*((1ll<<l)+(1ll<<r)-3);
FOR(cnt,0,l+r-4) if((res+cnt)%2==0){
// printf("l=%d,r=%d,cnt=%d,res+cnt=%d\n",l,r,cnt,res+cnt);
FOR(i,0,max(l,r)) FOR(j,0,min(cnt,2*i)) f[i][j][0]=f[i][j][1]=0;
f[0][0][0]=1;
FOR(i,1,max(l,r)) FOR(j,0,min(cnt,2*(i-1))){
FOR(k,0,1) FOR(a,0,1) FOR(b,0,1){
if(i>=l-1 && a==1) continue;
if(i>=r-1 && b==1) continue;
if((k+a+b)%2==((res+cnt)>>i)%2) f[i][j+a+b][(k+a+b)/2]+=f[i-1][j][k];
}
// printf("f[%d][%d][0]=%lld,f[%d][%d][1]=%lld\n",i-1,j,f[i-1][j][0],i-1,j,f[i-1][j][1]);
}
ans+=f[max(l,r)][cnt][0];
}
}
return ans;
}
int main(){
scanf("%lld",&n);
printf("%lld\n",solve1()+solve2());
}

CF750G New Year and Binary Tree Paths(DP)的更多相关文章

  1. LeetCode 257. Binary Tree Paths (二叉树路径)

    Given a binary tree, return all root-to-leaf paths. For example, given the following binary tree: 1 ...

  2. LeetCode OJ:Binary Tree Paths(二叉树路径)

    Given a binary tree, return all root-to-leaf paths. For example, given the following binary tree: 1 ...

  3. LeetCode 257. Binary Tree Paths(二叉树根到叶子的全部路径)

    Given a binary tree, return all root-to-leaf paths. Note: A leaf is a node with no children. Example ...

  4. LeetCode Binary Tree Paths(简单题)

    题意: 给出一个二叉树,输出根到所有叶子节点的路径. 思路: 直接DFS一次,只需要判断是否到达了叶子,是就收集答案. /** * Definition for a binary tree node. ...

  5. [CF750G] New Year and Binary Tree Paths

    目录 简单的 组合的 题目链接 简单的 设从节点\(x\)开始不断往左儿子走h-1步,则编号和为\(x\sum_{i=0}^{h-1}2^i=x(2^h-1)\). 若倒数第\(i\)步走向的是右儿子 ...

  6. zoj 3965 Binary Tree Restoring(搜索)

    Binary Tree Restoring Time Limit: 1 Second      Memory Limit: 65536 KB      Special Judge Given two ...

  7. Binary Tree Traversals(HDU1710)二叉树的简单应用

    Binary Tree Traversals Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/O ...

  8. LeetCode算法题-Binary Tree Paths(Java实现-3种解法)

    这是悦乐书的第199次更新,第206篇原创 01 看题和准备 今天介绍的是LeetCode算法题中Easy级别的第62题(顺位题号是257).给定二叉树,返回所有根到叶路径.例如: 输入: 1 / \ ...

  9. Palindromic Paths(DP)

    描述 Given an N×N grid of fields (1≤N≤500), each labeled with a letter in the alphabet. For example: A ...

随机推荐

  1. CocoaPods安装和使用201712

    CocoaPods安装使用详解 2017.12 首先,很有必要了解一下CocoaPods.Ruby和RubyGems,以及它们之间的关系. CocoaPods是第三方库的辅助管理工具,依赖于Ruby. ...

  2. dedecmsV5.7 任意文件上传漏洞修复

    问题文件:在/include/dialog/select_soft_post.php文件 解决方法: 先找到这个文件/include/dialog/select_soft_post.php,找到大概7 ...

  3. C# Excel 读取导入数据库

    使用Aspose.Cells组件. 表格第一行为表头合并,第二行为数据名称,从第三行开始数据. if (xtraOpenFileDialog1.ShowDialog() == DialogResult ...

  4. SpringBoot开发准备工作,保存备用,

    application.properties server.port=8080 spring.thymeleaf.prefix = classpath:/static/ spring.thymelea ...

  5. Java之Calendar类

    Calendar类概述 java.util.Calendar 是日历类,在Date后出现,替换掉了许多Date的方法.该类将所有可能用到的时间信息封装为静态成员变量,方便获取.日历类就是方便获取各个时 ...

  6. vue使用--vuex快速学习与使用

    什么是vuex? Vuex核心概念 Vuex安装与使用 1.安装 2.目录结构与vuex引入 3.store中变量的定义.管理.派生(getter) 4.vuex辅助函数的使用说明 Vuex刷新数据丢 ...

  7. opencv-python图像二值化函数cv2.threshold函数详解及参数cv2.THRESH_OTSU使用

    cv2.threshold()函数的作用是将一幅灰度图二值化,基本用法如下: #ret:暂时就认为是设定的thresh阈值,mask:二值化的图像 ret,mask = cv2.threshold(i ...

  8. 关于Oracle数据库的rownum应用

    它是Oracle系统顺序分配为从查询返回的行的编号,返回的第一行分配的是1,第二行是2,以此类推,这个伪字段可以用于限制查询返回的总行数,而且rownum不能以任何表的名称作为前缀. 如以下语句将无法 ...

  9. Spring3:spring的事务操作

    三.事务操作 1.导包 2. jdbc模板与开源连接池(DBCP与C3P0) 2.1DBCP 2.2C3P0 :: 2.3.抽取配置到属性文件   定义一个属性文件  在Spring的配置文件中引入属 ...

  10. 我的计划任务 --- 实现市电停电安全关闭群辉,Windows, Linux等设备

    有一次突然停电,我的群辉DS218+ 的一块硬盘出现故障了,让我担心我的数据安全,其实我是有UPS, 不是在线式的,然后就想如何实现停电自动关机呢? 经过半天的了解,其实群辉支持telnet协议,于是 ...