CF1485C Floor and Mod 题解
题意:
多组询问。给定 x,y ,求 1≤a≤x,1≤b≤y 且 ⌊ab⌋=amodb 的(a,b) 个数。
数据范围:
1≤Q≤100, 1≤x,y≤109 。
这道题有点麻烦 QAQ
先推一波柿子
⌊ab⌋=amodb⌊ab⌋=a−⌊ab⌋×b⌊ab⌋=ab+1显然 b+1∣a ,同时因为 amodb<b
所以 ab+1<b⇒a<b2+b⇒a∈[0,b2+b−1]
则满足条件的 a 的个数为 ⌊min{x,b2+b−1}b+1⌋
答案就是
y∑b=1⌊min{x,b2+b−1}b+1⌋但是这个东西好像不好用数论分块
仔细观察,当 b≥√x−ϵ,ϵ∈N 时,柿子是这样的
y∑b=1⌊xb+1⌋其实写成 y+1∑b=2⌊xb⌋ 可以更明显地看出这是个数论分块的板子
对于 b<√x−ϵ,ϵ∈N 的情况,直接暴力枚举即好了
时间复杂度 O(Q√x)
代码:
cpp
#include <iostream>
#include <string>
#include <vector>
#include <algorithm>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <iomanip>
#include <random>
using namespace std;
#define int long long
#define INF 0x3f3f3f3f3f3f3f3f
#define N (int)()
signed main()
{
ios::sync_with_stdio(0);
cin.tie(0);cout.tie(0);
// freopen("check.in","r",stdin);
// freopen("check.out","w",stdout);
int Q,a,b,x,y,res;
cin >> Q;
while(Q--)
{
cin >> x >> y;
a=1,b=1,res=0;
for(; b*b+b-1<x&&b<=y; b++)
res+=(b*b+b-1)/(b+1);
for(int l=b+1,r; l<=min(x,y+1); l=r+1)
{
r=min(x/(x/l),y+1);
res+=x/l*(r-l+1);
}
cout << res << '\n';
}
return 0;
}