BZOJ 3529: [Sdoi2014]数表 [莫比乌斯反演 树状数组]
3529: [Sdoi2014]数表
Time Limit: 10 Sec Memory Limit: 512 MB
Submit: 1399 Solved: 694
[Submit][Status][Discuss]
Description
有一张N×m的数表,其第i行第j列(1 < =i < =礼,1 < =j < =m)的数值为
能同时整除i和j的所有自然数之和。给定a,计算数表中不大于a的数之和。
Input
输入包含多组数据。
输入的第一行一个整数Q表示测试点内的数据组数,接下来Q行,每行三个整数n,m,a(|a| < =10^9)描述一组数据。
Output
对每组数据,输出一行一个整数,表示答案模2^31的值。
Sample Input
4 4 3
10 10 5
Sample Output
148
HINT
1 < =N.m < =10^5 , 1 < =Q < =2×10^4
Source
一个位置的答案就是gcd(i,j)的约数和
一个个单独算不好优化不行,从gcd(i,j)的取值方面考虑,因为它的取值就是1...n
设F(i)为i的约数和,f(i)为gcd(x,y)=i的个数,那么答案就是ΣF(i)*f(i)
f(i)上两题用到过,反演后f(i)=Σ{i|d} miu(d/i)*(n/d)*(m/d)
d和i的取值范围相同,可以得到如下式子
现在我们只需要求出g(i)=的前缀和 这个问题就能在O(√n)的时间内出解
F(i)是约数和函数,可以通过线性筛计算,或者直接nlogn暴力枚举倍数,速度差不多
然后和上一题一样,暴力枚举每个i更新它的倍数
那么a的限制如何处理?考虑离线
我们发现对答案有贡献的i只有F(i)<=a的i 那么我们将询问按照a从小到大排序 将F(i)从小到大排序 每次询问将<=a的F(i)按照枚举倍数更新的方式插入 用树状数组来维护g的前缀和 这样枚举倍数logn,每个倍数插入树状数组logn,总共nlog^2n
本题取模有一种好厉害的做法:自然溢出int,最后&0x7FFFFFFF
复杂度O(nlog^2n+q√nlogn)
注意:排序[1,N)的话是sort(a+1,a+N),不要a+N+1.......
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
typedef long long ll;
const int N=1e5+;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-; c=getchar();}
while(c>=''&&c<=''){x=x*+c-''; c=getchar();}
return x*f;
}
struct ques{
int n,m,a,id;
bool operator <(const ques &r)const{return a<r.a;}
}q[N];
int n,m;
int notp[N],p[N],mu[N],minfac[N],t1[N],t2[N];
struct data{
int s,i;
bool operator <(const data &r)const{return s<r.s;}
}sf[N];
void sieve(){
for(int i=;i<N;i++) sf[i].i=i;
mu[]=;
sf[].s=;
for(int i=;i<N;i++){
if(!notp[i]){
p[++p[]]=i,mu[i]=-;
minfac[i]=i;
sf[i].s=i+;
t1[i]=i+;
t2[i]=i;
}
for(int j=,k;j<=p[]&&(k=i*p[j])<N;j++){
notp[i*p[j]]=;
minfac[k]=p[j];
if(i%p[j]==){
mu[i*p[j]]=;
t2[k]=t2[i]*p[j];
t1[k]=t1[i]+t2[k];
sf[k].s=sf[i].s/t1[i]*t1[k];
break;
}
mu[i*p[j]]=-mu[i];
t1[k]=+p[j];
t2[k]=p[j];
sf[k].s=sf[i].s*sf[p[j]].s;
}
}
} int g[N];
inline int lowbit(int x){return x&-x;}
inline void add(int p,int v){for(int i=p;i<N;i+=lowbit(i)) g[i]+=v;}
inline int sum(int p){
int ret=;
for(int i=p;i;i-=lowbit(i)) ret+=g[i];
return ret;
}
void ins(int x){
for(int d=sf[x].i;d<N;d+=sf[x].i) add(d,sf[x].s*mu[d/sf[x].i]);
}
int cal(int n,int m){
int ans=,r=;
if(n>m) swap(n,m);
for(int i=;i<=n;i=r+){
r=min(n/(n/i),m/(m/i));
ans+=(sum(r)-sum(i-))*(n/i)*(m/i);
}
return ans;
}
int ans[N];
int main(int argc, const char * argv[]){
sieve();
sort(sf+,sf+N);//!!!!!
int T=read();
for(int i=;i<=T;i++) q[i].n=read(),q[i].m=read(),q[i].a=read(),q[i].id=i;
sort(q+,q++T);
int now=;
for(int i=;i<=T;i++){
int a=q[i].a;
for(;sf[now].s<=a;now++) ins(now);
ans[q[i].id]=cal(q[i].n,q[i].m)&0x7FFFFFFF;
}
for(int i=;i<=T;i++) printf("%d\n",ans[i]);
return ;
}
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
using namespace std;
typedef long long ll;
const int N=1e5+;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-; c=getchar();}
while(c>=''&&c<=''){x=x*+c-''; c=getchar();}
return x*f;
}
struct ques{
int n,m,a,id;
bool operator <(const ques &r)const{return a<r.a;}
}q[N];
int n,m;
int notp[N],p[N],mu[N];
struct data{
int s,i;
bool operator <(const data &r)const{return s<r.s;}
}sf[N];
void sieve(){
mu[]=;
for(int i=;i<=;i++){
if(!notp[i]){
p[++p[]]=i,mu[i]=-;
}
for(int j=,k;j<=p[]&&(k=i*p[j])<=;j++){
notp[k]=;
if(i%p[j]==){
mu[k]=;
break;
}
mu[k]=-mu[i];
}
}
for(int i=;i<=;i++){
sf[i].i=i;
for(int j=;j*i<=;j++) sf[i*j].s+=i;
}
} int g[N];
inline int lowbit(int x){return x&-x;}
inline void add(int p,int v){for(int i=p;i<N;i+=lowbit(i)) g[i]+=v;}
inline int sum(int p){
int ret=;
for(int i=p;i;i-=lowbit(i)) ret+=g[i];
return ret;
}
inline void ins(int x){
for(int d=sf[x].i;d<=;d+=sf[x].i) add(d,sf[x].s*mu[d/sf[x].i]);
}
int cal(int n,int m){
int ans=,r=;
if(n>m) swap(n,m);
for(int i=;i<=n;i=r+){
r=min(n/(n/i),m/(m/i));
ans+=(sum(r)-sum(i-))*(n/i)*(m/i);
}
return ans;
}
int ans[N];
int main(int argc, const char * argv[]){
sieve();
sort(sf+,sf+);
int T=read();
for(int i=;i<=T;i++) q[i].n=read(),q[i].m=read(),q[i].a=read(),q[i].id=i;
sort(q+,q++T);
int now=;
for(int i=;i<=T;i++){
int a=q[i].a;
for(;now<=&&sf[now].s<=a;now++) ins(now);
ans[q[i].id]=cal(q[i].n,q[i].m)&0x7FFFFFFF;
}
for(int i=;i<=T;i++) printf("%d\n",ans[i]);
return ;
}
BZOJ 3529: [Sdoi2014]数表 [莫比乌斯反演 树状数组]的更多相关文章
- BZOJ 3529 [Sdoi2014]数表 (莫比乌斯反演+树状数组+离线)
题目大意:有一张$n*m$的数表,第$i$行第$j$列的数是同时能整除$i,j$的所有数之和,求数表内所有不大于A的数之和 先是看错题了...接着看对题了发现不会做了...刚了大半个下午无果 看了Po ...
- BZOJ 3529 [Sdoi2014]数表 ——莫比乌斯反演 树状数组
$ans=\sum_{i=1}^n\sum_{j=1}^n\sigma(gcd(i,j))$ 枚举gcd为d的所有数得到 $ans=\sum_{d<=n}\sigma(d)*g(d)$ $g(d ...
- BZOJ 3259 [Sdoi2014]数表 (莫比乌斯反演 + 树状数组)
3529: [Sdoi2014]数表 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 2321 Solved: 1187[Submit][Status ...
- 【BZOJ3529】[Sdoi2014]数表 莫比乌斯反演+树状数组
[BZOJ3529][Sdoi2014]数表 Description 有一张N×m的数表,其第i行第j列(1 < =i < =礼,1 < =j < =m)的数值为能同时整除i和 ...
- BZOJ3529: [Sdoi2014]数表(莫比乌斯反演 树状数组)
题意 题目链接 Sol 首先不考虑\(a\)的限制 我们要求的是 \[\sum_{i = 1}^n \sum_{j = 1}^m \sigma(gcd(i, j))\] 用常规的套路可以化到这个形式 ...
- luogu3312 [SDOI2014]数表 (莫比乌斯反演+树状数组)
link \(\sum_{i=1}^n\sum_{j=1}^m[s(\gcd(i,j))\le a]s(\gcd(i,j))\) \(=\sum_{p=1}^ns(p)[s(p)\le a]\sum_ ...
- bzoj 3529 数表 莫比乌斯反演+树状数组
题目大意: 有一张N×m的数表,其第i行第j列(1 < =i < =礼,1 < =j < =m)的数值为能同时整除i和j的所有自然数之和.给定a,计算数表中不大于a的数之和. ...
- 【BZOJ3529】【莫比乌斯反演 + 树状数组】[Sdoi2014]数表
Description 有一张N×m的数表,其第i行第j列(1 < =i < =礼,1 < =j < =m)的数值为 能同时整除i和j的所有自然数之和.给定a,计算数表中不大于 ...
- BZOJ_3529_[Sdoi2014]数表_莫比乌斯反演+树状数组
Description 有一张 n×m 的数表,其第 i 行第 j 列(1 <= i <= n, 1 <= j <= m)的数值为 能同时整除 i 和 j 的所有自然数之和.给 ...
随机推荐
- 由浅入深学习ajax跨域(JSONP)问题
什么是跨域?说直白点就是获取别人网站上的内容.但这么说貌似又有点混淆,因为通常我们用ajax+php就可以获取别人网站的内容,来看下面这个例子. 来看看跨域的例子,jquery+ajax是不能跨域请求 ...
- 用JavaScript来实现链表LinkedList
本文版权归博客园和作者本人共同所有,转载和爬虫请注明原文地址. 写在前面 好多做web开发的朋友,在学习数据结构和算法时可能比较讨厌C和C++,上学的时候写过的也忘得差不多了,更别提没写过的了.但幸运 ...
- c++ map 使用
. 包含头文件: #include <map> 2. 构造函数: std::map<char,int> first; first[; first[; first[; first ...
- Aix/Linux下自动备份oracle数据库
曾经有个同事,来回操作开发和生产的数据库,结果误删了生产的数据库,那种心情我想不是一般人能理解的,虽然说oracle可以有方法还原,但并不是彻底的. 所以,在工作中,不管是开发还是维护,备份数据库是非 ...
- 工业物联网或系统集成中应用消息队列(ActiveMQ,C#的demo)的场景全面分析
1.[连载]<C#通讯(串口和网络)框架的设计与实现> 2.[开源]C#跨平台物联网通讯框架ServerSuperIO(SSIO)介绍 2.应用SuperIO(SIO)和开源跨平台物联网框 ...
- 每天一个设计模式-3 适配器模式(Adapteer)
每天一个设计模式-3 适配器模式(Adapteer) 1.现实中的情况 旧式电脑的硬盘是串口的,直接与硬盘连接,新硬盘是并口的,显然新硬盘不能直接连在电脑上,于是就有了转接线.好了,今天的学习主题出来 ...
- JS定时刷新页面及跳转页面
JS定时刷新页面及跳转页面 Javascript 返回上一页1. Javascript 返回上一页 history.go(-1), 返回两个页面: history.go(-2); 2. history ...
- JavaScript学习总结——我所理解的JavaScript闭包
一.闭包(Closure) 1.1.什么是闭包? 理解闭包概念: a.闭包是指有权限访问另一个函数作用域的变量的函数,创建闭包的常见方式就是在一个函数内部创建另一个函数,也就是创建一个内部函数,创建一 ...
- kmdjs和循环依赖
循环依赖 循环依赖是非常必要的,有的程序写着写着就循环依赖了,可以提取出一个对象来共同依赖解决循环依赖,但是有时会破坏程序的逻辑自封闭和高内聚.所以没解决好循环依赖的模块化库.框架.编译器都不是一个好 ...
- Android开发学习—— ContentProvider内容提供者
* 应用的数据库是不允许其他应用访问的* 内容提供者的作用就是让别的应用访问到你的数据库.把私有数据暴露给其他应用,通常,是把私有数据库的数据暴露给其他应用. Uri:包含一个具有一定格式的字符串的对 ...