• codevs 2606 约数和问题


    题目描述 Description

    Smart最近沉迷于对约数的研究中。

    对于一个数X,函数f(X)表示X所有约数的和。例如:f(6)=1+2+3+6=12。对于一个X,Smart可以很快的算出f(X)。现在的问题是,给定两个正整数X,Y(X<Y),Smart希望尽快地算出f(X)+f(X+1)+……+f(Y)的值,你能帮助Smart算出这个值吗?

    输入描述 Input Description

    输入文件仅一行,两个正整数X和Y(X<Y),表示需要计算f(X)+f(X+1)+……+f(Y)。

    输出描述 Output Description

    输出只有一行,为f(X)+f(X+1)+……+f(Y)的值。

     

      首先,对于区间$[l,r]$显然可以拆成$[1,r]$与$[1,l-1]$的差。然后,由于求的是区间的因数和,那么一个显然的想法就是枚举约数$x$,若区间内有$y$个数有$x$这个约数,那么$ans$就要加上$x*y$,对于区间$[1,n]$也就是$lfloor n/i floor *i$。

      这样一来,我们就把问题转化成了求这个式子的值:

    $$sum_{i=1}^{n}lfloor n/i floor * i$$

      由于$lfloor n/i floor$只有$sqrt{n}$种取值,就可以对于$i le sqrt{n}$的暴力算一下$lfloor n/i floor * i$,同时算一下$lfloor n/x floor =i$的$x$的范围,统计一下即可。

      下面贴代码:

     1 #include<cstdio>
     2 #include<cmath>
     3 
     4 using namespace std;
     5 typedef long long llg;
     6 
     7 llg work(int x){
     8     llg ans=0,la=x,gi=(llg)sqrt(x);
     9     for(llg i=1,g;i<=gi;i++){
    10         g=x/i;
    11         ans+=i*g+(g+1+la)*(la-g)/2*(i-1);
    12         la=x/i;
    13     }
    14     ans+=(gi+1+la)*(la-gi)/2*gi;
    15     return ans;
    16 }
    17 
    18 int main(){
    19     int l,r;
    20     scanf("%d %d",&l,&r);
    21     printf("%lld",work(r)-work(l-1));
    22 }
  • 相关阅读:
    无监督学习
    监督学习
    cmd
    oj1026
    oj1025
    使用虚函数的不同模式
    hdu1166:敌兵布阵(树状数组或线段树)
    传纸条(动态规划)
    SDUT 1266 出栈序列统计(卡特兰数)
    HDU 5063 Operation the Sequence
  • 原文地址:https://www.cnblogs.com/lcf-2000/p/5861108.html
Copyright © 2020-2023  润新知