【SDOI2014】【BZOJ3529】数表

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

2

4 4 3

10 10 5

Sample Output

20

148

HINT

1 < =N.m < =10^5 , 1 < =Q < =2×10^4

Source

Round 1 Day 1

在反演上的处理和DzyLovesMath1是类似的属于同一类题目
问题在于询问有10^4个所以要BIT维护一下前缀和
推公式过程懒得放了…贴个Po姐课件的图吧
这里写图片描述
这里写图片描述
(从未见过如此丧病的反演..反演还要加数据结构T_T

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cmath>
#include<algorithm>
#define MAXN 100010
#define lowbit(x)   (x&(-x))
#define LL long long
#define MAXINT (0x7fffffff)
#define GET (ch>='0'&&ch<='9')
using namespace std;
int T,maxn;
int c[MAXN];
bool not_prime[MAXN];
int prime[MAXN],mu[MAXN]={0,1},top;
int ans[MAXN];
struct Query
{
    int n,m,a,id;
    bool operator <(const Query& t)const    {return a<t.a;}
}q[MAXN];
struct num
{
    int a,b;
    bool operator <(const num& x)const  {return a==x.a?b<x.b:a<x.a;}
}f[MAXN];
void init()
{
    for (int i=2;i<=maxn;i++)
    {
        if (!not_prime[i])  prime[++top]=i,mu[i]=-1;
        for (int j=1;j<=top&&i*prime[j]<=maxn;j++)
        {
            not_prime[i*prime[j]]=1;mu[i*prime[j]]=-mu[i];
            if (i%prime[j]==0)  {mu[i*prime[j]]=0;break;}
        }
    }
    for (int i=1;i<=maxn;i++)
    {
        for (int j=i;j<=maxn;j+=i)  f[j].a+=i;
        f[i].b=i;
    }
}
void add(int x,int delta)   {for (int i=x;i<=maxn;i+=lowbit(i)) c[i]+=delta;}
int query(int x)
{
    int ret=0;
    for (int i=x;i;i-=lowbit(i))    ret+=c[i];
    return ret;
}
void in(int &x)
{
    char ch=getchar();x=0;int flag=1;
    while (!GET)    flag=(ch=='-')?-1:1,ch=getchar();
    while (GET) x=x*10+ch-'0',ch=getchar();x*=flag;
}
void calc(int x)
{
    int id=q[x].id,n=q[x].n,m=q[x].m,last=0,t=min(n,m);
    for (int i=1;i<=t;i=last+1)
    {
        last=min(n/(n/i),m/(m/i));
        ans[id]+=(n/i)*(m/i)*(query(last)-query(i-1));
    }
}
int main()
{
    in(T);int now=0;
    for (int i=1;i<=T;i++)  in(q[i].n),in(q[i].m),in(q[i].a),q[i].id=i,maxn=max(maxn,max(q[i].n,q[i].m));
    init();sort(q+1,q+T+1);sort(f+1,f+maxn+1);
    for (int i=1;i<=T;i++)
    {
        while (now+1<=maxn&&f[now+1].a<=q[i].a)
        {
            now++;
            for (int j=f[now].b;j<=maxn;j+=f[now].b)    add(j,f[now].a*mu[j/f[now].b]);
        }
        calc(i);
    }
    for (int i=1;i<=T;i++)  printf("%d\n",ans[i]&MAXINT);
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值