0x00 前置芝士

数位dp考试里出现的小神题??

显然考场会选择打表找规律。

数位dp + 矩阵快速幂


0x01 题目描述

给定正整数 \(n\),现有如下方程 \(x \bigoplus 3x = 2x\),其中 \(\bigoplus\) 表示按位异或。

任务如下:

  1. 求出小于等于 \(n\) 的正整数中,有多少个数是该方程的解
  2. 求出小于等于 \(2^n\) 的正整数中,有多少个数是该方程的解,答案对 \(10^9 + 7\) 取模

0x02 分析

第一问

试证明满足 \(x \bigoplus 2 \times x = 3 \times x\) 的 \(x\),二进制拆分数列里没有相邻的 \(1\)。

条件可化为 \(x \bigoplus 2 \times x = x + 2 \times x\)。

若有相邻的 \(1\),二倍后即错位相加,定会产生多余进位,得不到右边的答案,故矛盾。故原命题成立。

例:

0 0 1 1
0 1 1 0

推论: 设 \(f(a, b)(b \in \{0, 1\})\) 表示第 \(a - 1\) 一位为 \(b\) 的数中共有多少个满足条件的数。

则 \(f(a, 0) = f(a - 1, 1) + f(a - 2, 0)\),\(f(a, 1) = f(a - 1, 0)\)。证明显然。

第二问

试证明小于 \(2 ^ n\) 的满足 \(x \bigoplus 2 \times x = 3 \times x\) 的 \(x\) 的个数为斐波那契数列第 \(n\) 项。

记 \(g(a) = f(a, 0) + f(a, 1)\)。利用第一问结论推论,推导如下。

\(g(a) = f(a, 0) + f(a, 1)\)

\(g(a) = f(a - 1, 1) + f(a - 1, 0) + f(a - 1, 0)\)

\(g(a) = g(a - 1) + f(a - 2, 1) + f(a - 2, 0)\)

\(g(a) = g(a - 1) + g(a - 2)\)


0x04 具体实现

#include <cstdio>
#include <cstring>
using namespace std; typedef long long LL;
const int MAXL = 105;
const int MAXN = 4;
const int mod = 1e9 + 7;
int flag[MAXL], a[MAXL];
LL dp[MAXL][MAXL][2][2]; struct Matrix {
int n, m;
LL mp[MAXN][MAXN];
Matrix() { memset(mp, 0, sizeof mp); }
Matrix operator * (const Matrix &x) const {
Matrix ans;
ans.n = n;
ans.m = x.m;
for (int i = 1; i <= ans.n; i++)
for (int j = 1; j <= ans.m; j++)
for (int k = 1; k <= m; k++)
ans.mp[i][j] = (ans.mp[i][j] + (mp[i][k] * x.mp[k][j]) % mod) % mod;
return ans;
}
}; Matrix Quick_pow(Matrix a, LL x) {
Matrix ans;
ans.n = 2;
ans.m = 2;
ans.mp[2][2] = 1;
ans.mp[1][1] = 1;
ans.mp[1][2] = 0;
ans.mp[2][1] = 0;
while (x) {
if (x & 1)
ans = ans * a;
a = a * a;
x >>= 1;
}
return ans;
} LL dfs(int p, int last, bool k, bool limit, bool t) {
if(p <= 0)
return !t && k;
if(!limit && dp[p][last][t][k] != -1)
return dp[p][last][t][k];
int up = limit ? flag[p] : 1;
LL ans = 0;
for(int i = 0; i <= up; i++)
ans = (ans + dfs(p - 1, i, t || last == -1 || (k && i == 1 && last == 0)
|| (k && i == 0 && last == 0) || (k && i == 0 && last == 1), limit && (i == up), (t && !i)));
if(!limit)
dp[p][last][t][k] = ans;
return ans;
} LL Query(LL x) {
int len = 0;
while(x) {
flag[++len] = (x & 1);
x >>= 1;
}
return dfs(len, -1, true, true, true);
} int main() {
memset(dp, -1, sizeof dp);
int T;
scanf ("%d", &T);
while(T--) {
LL n;
scanf ("%lld", &n);
printf("%lld\n", Query(n));
Matrix A, cur, ans;
A.n = 2, A.m = 2;
A.mp[1][1] = 0;
A.mp[1][2] = 1;
A.mp[2][1] = 1;
A.mp[2][2] = 1;
ans = Quick_pow(A, n);
cur.n = 1, cur.m = 2;
cur.mp[1][1] = 1;
cur.mp[1][2] = 1;
ans = cur * ans;
printf("%lld\n", ans.mp[1][2] % mod);
}
return 0;
}

Solution -「校内题」Xorequ的更多相关文章

  1. Solution -「校内题」矩阵求和

    Description 共 \(T\) 组数据.对于每组数据,给定 \(a, b, n\),求 \(\sum_{i = 1}^{n} \sum_{j = 1}^{n} \gcd(a^i - b^i, ...

  2. Solution -「ARC 104E」Random LIS

    \(\mathcal{Description}\)   Link.   给定整数序列 \(\{a_n\}\),对于整数序列 \(\{b_n\}\),\(b_i\) 在 \([1,a_i]\) 中等概率 ...

  3. Solution -「ARC 101D」「AT4353」Robots and Exits

    \(\mathcal{Description}\)   Link.   有 \(n\) 个小球,坐标为 \(x_{1..n}\):还有 \(m\) 个洞,坐标为 \(y_{1..m}\),保证上述坐标 ...

  4. Solution -「国家集训队」「洛谷 P2619」Tree I

    \(\mathcal{Description}\)   Link.   给一个 \(n\) 个点 \(m\) 条边的带权无向图,边有权值和黑白颜色,求恰选出 \(K\) 条白边构成的最小生成树.    ...

  5. Solution -「ARC 104C」Fair Elevator

    \(\mathcal{Description}\)   Link.   数轴从 \(1\sim 2n\) 的整点上有 \(n\) 个闭区间.你只知道每个区间的部分信息(可能不知道左或右端点,或者都不知 ...

  6. Solution -「基环树」做题记录

    写的大多只是思路,比较简单的细节和证明过程就不放了,有需者自取. 基环树简介 简单说一说基环树吧.由名字扩展可得这是一类以环为基础的树(当然显然它不是树. 通常的表现形式是一棵树再加一条非树边,把图画 ...

  7. Solution -「CF 1342E」Placing Rooks

    \(\mathcal{Description}\)   Link.   在一个 \(n\times n\) 的国际象棋棋盘上摆 \(n\) 个车,求满足: 所有格子都可以被攻击到. 恰好存在 \(k\ ...

  8. Solution -「CTS 2019」「洛谷 P5404」氪金手游

    \(\mathcal{Description}\)   Link.   有 \(n\) 张卡牌,第 \(i\) 张的权值 \(w_i\in\{1,2,3\}\),且取值为 \(k\) 的概率正比于 \ ...

  9. Solution -「BZOJ 3812」主旋律

    \(\mathcal{Description}\)   Link.   给定含 \(n\) 个点 \(m\) 条边的简单有向图 \(G=(V,E)\),求 \(H=(V,E'\subseteq E)\ ...

随机推荐

  1. 《Mybatis 手撸专栏》第7章:SQL执行器的定义和实现

    作者:小傅哥 博客:https://bugstack.cn - <手写Mybatis系列> 一.前言 为什么,要读框架源码? 因为手里的业务工程代码太拉胯了!通常作为业务研发,所开发出来的 ...

  2. 多级级联数据的展示-vue递归组件

    如果采用普通的for循环方式,没办法确认数据到底有几层,要写几个for循环,所以想到了递归的方法. 那么在vue里然后实现呢? vue递归组件(组件中使用自己) 父组件中把数据以props形式传给子组 ...

  3. 手脱PESpin壳【06.exe】

    1.查壳 2.LoradPE工具检查 一方面可以用LoradPE工具查看重定位,另一方面也可获取一些详细信息 3.查找OEP ①未发现pushad 开始未发现pushad,进行单步步入,很快就能找到p ...

  4. 渗透:winpcap

    winpcap(windows packet capture)是windows平台下一个免费,公共的网络访问系统.开发winpcap这个项目的目的在于为win32应用程序提供访问网络底层的能力.它提供 ...

  5. Python搜索书名获取整本资源_笔趣阁

    前言 偶然一天把某项目文档传到手机上,用手机自带的阅读器方便随时拿出来查阅.看着我那好久没点开的阅读器,再看着书架上摆着几本不知道是多久之前导入的小说. 闭上眼,我仿佛看到那时候的自己.侧躺着缩在被窝 ...

  6. java对象与Json字符串之间的转化

    public class Test { public static void main(String[] args) { // 实现java对象与Json字符串之间的转化 // 1. Person对象 ...

  7. Kubernetes Job Controller 原理和源码分析(二)

    概述程序入口Job controller 的创建Controller 对象NewController()podControlEventHandlerJob AddFunc DeleteFuncJob ...

  8. python常用标准库(os系统模块、shutil文件操作模块)

    常用的标准库 系统模块 import os 系统模块用于对系统进行操作. 常用方法 os模块的常用方法有数十种之多,本文中只选出最常用的几种,其余的还有权限操作.文件的删除创建等详细资料可以参考官方文 ...

  9. USB机械键盘改蓝牙键盘

    手里有两把机械键盘,一个是IKBC 87键,一个是IKBC POKER II 60键,由于买的比较早,两把键盘均为USB的,使用起来桌面线比较多,碍事,于是开始研究如何改成蓝牙键盘. 首先说一下USB ...

  10. Java常用类-包装类

    包装类 ​ Java中的基本类型功能简单,不具备对象的特性,为了使基本类型具备对象的特性,所以出现了包装类,就可以像操作对象一样操作基本类型数据;包装类不是为了取代基本数据类型,而是在数据类型需要使用 ...