【SDOI2014】数表
题面
题解
不管$a$的限制
我们要求的东西是:($\sigma(x)$是$x$的约数个数和)
$ \sum_{i=1}^n\sum_{j=1}^m\sigma(gcd(i,j)) $
设$f(x)=\sigma(x)$,则我们可以找到一个$g$使得$f=1*g$,那么$g=\mu*f$
所以$g(x)=\sum_{d|x}\mu(d)\sigma(\frac xd)$
带入原式得:
$ \sum_{i=1}^n\sum_{j=1}^m\sigma(gcd(i,j)) \\ =\sum_{d=1}^n g(d)\left\lfloor\frac nd\right\rfloor \left\lfloor\frac md\right\rfloor \\ =\sum_{d=1}^n \left\lfloor\frac nd\right\rfloor \left\lfloor\frac md\right\rfloor \sum_{x|d}\mu(x)\sigma(\frac dx) $
我们可以筛出$g(x)$的值来$O(\sqrt n)$回答
但是现在有$a$的限制,就可以离线将询问排序,将$\sigma$排序,加入相对应的位置就可以了。
代码
#include<cstdio>
#include<cstring>
#include<cctype>
#include<climits>
#include<algorithm>
#define RG register
#define file(x) freopen(#x".in", "r", stdin);freopen(#x".out", "w", stdout);
#define clear(x, y) memset(x, y, sizeof(x))
inline int read()
{
int data = 0, w = 1; char ch = getchar();
while(ch != '-' && (!isdigit(ch))) ch = getchar();
if(ch == '-') w = -1, ch = getchar();
while(isdigit(ch)) data = data * 10 + (ch ^ 48), ch = getchar();
return data * w;
}
const int maxn(100010);
bool not_prime[maxn];
int prime[maxn], cnt, n, T, ans[maxn];
int mu[maxn], sig[maxn], sumd[maxn], powd[maxn];
struct qry { int n, m, a, id; } q[maxn];
int p[maxn], c[maxn];
inline bool operator < (const qry &a, const qry &b) { return a.a < b.a; }
void init(int N)
{
not_prime[1] = true; sig[1] = mu[1] = p[1] = 1;
for(RG int i = 2; i <= N; i++)
{
p[i] = i;
if(!not_prime[i]) prime[++cnt] = i, mu[i] = -1,
sig[i] = i + 1, sumd[i] = i + 1, powd[i] = i;
for(RG int j = 1; j <= cnt && i * prime[j] <= N; j++)
{
not_prime[i * prime[j]] = true;
if(i % prime[j])
{
mu[i * prime[j]] = -mu[i];
sig[i * prime[j]] = sig[i] * sig[prime[j]];
sumd[i * prime[j]] = prime[j] + 1;
powd[i * prime[j]] = prime[j];
}
else
{
powd[i * prime[j]] = powd[i] * prime[j];
sumd[i * prime[j]] = sumd[i] + powd[i * prime[j]];
sig[i * prime[j]] = sig[i] / sumd[i] * sumd[i * prime[j]];
break;
}
}
}
}
inline bool cmp(int a, int b) { return sig[a] < sig[b]; }
void add(int x, int v) { while(x <= n) c[x] += v, x += x & -x; }
int query(int x) { int ans = 0; while(x) ans += c[x], x -= x & -x; return ans; }
int solve(int x, int y)
{
int now = 0, pre = 0, ans = 0;
for(RG int i = 1, j; i <= x; i = j + 1)
{
j = std::min(x / (x / i), y / (y / i));
now = query(j); ans += (x / i) * (y / i) * (now - pre);
pre = now;
}
return ans;
}
int main()
{
T = read();
for(RG int i = 1; i <= T; i++)
{
q[i] = (qry) {read(), read(), read(), i};
if(q[i].n > q[i].m) std::swap(q[i].n, q[i].m);
n = std::max(n, q[i].n);
}
init(n); std::sort(q + 1, q + T + 1);
std::sort(p + 1, p + n + 1, cmp);
for(RG int i = 1, j = 1; i <= T; i++)
{
for(; j <= n && sig[p[j]] <= q[i].a; ++j)
for(RG int k = p[j]; k <= n; k += p[j])
if(mu[k / p[j]]) add(k, sig[p[j]] * mu[k / p[j]]);
ans[q[i].id] = solve(q[i].n, q[i].m);
}
for(RG int i = 1; i <= T; i++)
{
if(ans[i] < 0) ans[i] += INT_MAX, ++ans[i];
printf("%d\n", ans[i]);
}
return 0;
}
【SDOI2014】数表的更多相关文章
- BZOJ 3529: [Sdoi2014]数表 [莫比乌斯反演 树状数组]
3529: [Sdoi2014]数表 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 1399 Solved: 694[Submit][Status] ...
- 【BZOJ 3529】 [Sdoi2014]数表 (莫比乌斯+分块+离线+树状数组)
3529: [Sdoi2014]数表 Description 有一张N×m的数表,其第i行第j列(1 < =i < =礼,1 < =j < =m)的数值为能同时整除i和j的所有 ...
- BZOJ 3259 [Sdoi2014]数表 (莫比乌斯反演 + 树状数组)
3529: [Sdoi2014]数表 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 2321 Solved: 1187[Submit][Status ...
- 洛咕3312 [SDOI2014]数表
洛咕3312 [SDOI2014]数表 终于独立写出一道题了...真tm开心(还是先写完题解在写的) 先无视a的限制,设\(f[i]\)表示i的约数之和 不妨设\(n<m\) \(Ans=\su ...
- 【BZOJ3529】[Sdoi2014]数表 莫比乌斯反演+树状数组
[BZOJ3529][Sdoi2014]数表 Description 有一张N×m的数表,其第i行第j列(1 < =i < =礼,1 < =j < =m)的数值为能同时整除i和 ...
- 洛谷 P3312 [SDOI2014]数表 解题报告
P3312 [SDOI2014]数表 题目描述 有一张\(N*M\)的数表,其第\(i\)行第\(j\)列(\(1\le i \le n\),\(1 \le j \le m\))的数值为能同时整除\( ...
- BZOJ3529 [Sdoi2014]数表 【莫比乌斯反演】
3529: [Sdoi2014]数表 Time Limit: 10 Sec Memory Limit: 512 MB Submit: 2151 Solved: 1080 [Submit][Status ...
- BZOJ[Sdoi2014]数表 莫比乌斯反演
[Sdoi2014]数表 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 2383 Solved: 1229[Submit][Status][Disc ...
- 洛谷P3312 - [SDOI2014]数表
Portal Solution 共\(T(T\leq2\times10^4)\)组测试数据.给出\(n,m(n,m\leq10^5),a(a\leq10^9)\),求\[ \sum_{i=1}^n\s ...
- bzoj [SDOI2014]数表 莫比乌斯反演 BIT
bzoj [SDOI2014]数表 莫比乌斯反演 BIT 链接 bzoj luogu loj 思路 \[ \sum\limits_{i=1}^{n}\sum\limits_{j=1}^{m}a*[f[ ...
随机推荐
- BZOJ 1041 圆上的整点 数学
题目链接: https://www.lydsy.com/JudgeOnline/problem.php?id=1041 题目大意:求一个给定的圆(x^2+y^2=r^2),在圆周上有多少个点的坐标是整 ...
- python中基于descriptor的一些概念(上)
@python中基于descriptor的一些概念(上) python中基于descriptor的一些概念(上) 1. 前言 2. 新式类与经典类 2.1 内置的object对象 2.2 类的方法 2 ...
- linux服务器部署nodejs项目
一.安装nodejs 1.去官网下载和自己系统匹配的文件: 英文网址:https://nodejs.org/en/download/ 中文网址:http://nodejs.cn/download/ 通 ...
- JSTL的下载和配置
1,首先在输入网址http://www.oracle.com/technetwork/java/jstl-137486.html 2,单击红色图标所示 3,单击Download 4,单击JSTL Im ...
- I、Q信号是如何产生的,I、Q信号复用的作用
接收机在中频部分实现模数变换和采样,采样后的信号和数字域的同频相乘,就可以得到基带的I.Q分量.在无线接口传输时,每一种使用特定的载波频率.码(扩频码和扰码)以及载波相对相位(I或Q)的信道都可以理解 ...
- HDU 2544最短路 (迪杰斯特拉算法)
传送门: http://acm.hdu.edu.cn/showproblem.php?pid=2544 最短路 Time Limit: 5000/1000 MS (Java/Others) Me ...
- Oracle特殊查询 行列倒转 分页
--查询工资最高的前三名 (分页的感觉)select * from(select * from emp order by sal desc) twhere rownum <=3--查询工资最高的 ...
- Web | JavaScript的引用数据类型强制转换类型
我在这里主要的想提下的是JavaScript中的引用类型进行强制转换类型.因为对于基本数据类型的变换大多都是雷同的,很容易熟知,但是引用数据类型有一点小插曲. JavaScript的引用类型主要为对象 ...
- new和delete的三种形式详解
一.new操作符.delete操作符 class String { public: String(const char *str="") { if(str== NULL) { da ...
- python使用findall正则匹配出所有符合条件的字符串
# -*- coding:utf-8 -*- import re mystr="qqq key:www.baidu.com<br>key:www.tengxun.com<b ...