思维+乘法原理
题意:
给出一个上限 ,和一个模数 ,计算符合下列条件的数列 的个数:
- 数列 的长度大于等于 。
- 数列 的最小值大于等于 ,最大值小于等于 ,且是严格单调递增的。
- 对于
用如下方式构造一个数列
, 要求构造的数列
是严格单调递增的。
题解:
参考官方题解:
首先定义一个函数
,表示
的二进制表示中数位为1的最高位是哪一位。比如
。由a是严格单调递增的条件我们可以知道,
。
然后我们利用b是严格单调递增的条件,
。可以知道
,如果
的话,那么异或之后这个最高位将会变成0,导致
。
同时我们也可以得出
。依次类推就可以得出
。
每一个 都是不同的,且是严格递增的。那么在符合条件的数列中,只有一个 或者没有 满足 。也就是只有一个或者没有 属于 。只要从每一个数段中选出一个数,或者不选,这样组成的数列从小到大排序都是合法的。每一段有 个选择。根据乘法原理就可求出总数,最后要减去每一个都不选形成的数列。
代码:
/**
* Author : Xiuchen
* Date : 2020-04-07-12.02.25
* Description : 乘法原理
*/
#pragma GCC optimize(1)
#pragma GCC optimize(2)
#pragma GCC optimize(3,"Ofast","inline")
#include<cstdio>
#include<cstring>
#include<string>
#include<vector>
#include<stack>
#include<queue>
#include<map>
#include<cmath>
#include<math.h>
#include<iostream>
#include<algorithm>
using namespace std;
typedef long long ll;
const int inf = 0x3f3f3f3f;
const ll INF = 0x3f3f3f3f3f3f3f3fLL;
//const int maxn = ;
int gcd(int a, int b){
return b ? gcd(b, a % b) : a;
}
int t;
ll d, m;
int main(){
cin >> t;
while(t--){
cin >> d >> m;
ll ans = 1 % m;
for(int i = 0; i <= 31; i++){
ll tmp = 1 << i;
if(tmp > d) break;
ll num = min((1ll << (i + 1)) - 1, d) - tmp + 2;
ans = ans * num % m;
}
cout << (ans - 1 + m) % m << endl;
}
return 0;
}