又是一道经典题.

1.学习了下O(n) 的做法。

//
// main.cpp
// bzoj2154
//
// Created by New_Life on 16/7/7.
// Copyright © 2016年 chenhuan001. All rights reserved.
// #include <iostream>
#include <string.h>
#include <stdio.h>
using namespace std; #define N 10001000
#define MOD 20101009 //--莫比乌斯反演函数--//
//说明:利用线性素数筛选顺便求了个mu
//注释部分为求从区间[1,b]和区间[1,d]中取两个数,互质对数O(n^0.5)
//复杂度:O(n)
int mu[N];
long long sum[N];
int prime[N];
bool mark[N]; void mobus()
{
int pcnt=;
memset(mark,,sizeof(mark));
mu[] = ;
for(int i=;i<N;i++)
{
if(mark[i] == )
{
prime[pcnt++] = i;
mu[i] = -;
}
for(int j=;j<pcnt && i*prime[j]<N;j++)
{
int tmp = i*prime[j];
mark[tmp] = ;
if( i%prime[j] == )
{
mu[tmp] = ;
break;
} mu[tmp] = mu[i]*-;
}
}
for(int i=;i<N;i++)
{
sum[i] += sum[i-]+(long long)mu[i]*i*i;
sum[i] %= MOD;
}
} long long gaobili(long long b,long long d)
{
if(b<=||d<=) return ;
long long m = min(b,d);
long long ans = ;
while(m>=)
{
long long tb = b/( b/m + )+;
long long td = d/( d/m + )+;
//前进的最大位置
long long tm = max(tb,td);
ans += (sum[m]-sum[tm-])*(((b/m+)*(b/m)/)%MOD)%MOD*(((d/m+)*(d/m)/)%MOD)%MOD ;
ans %= MOD;
m = tm-;
}
return ans;
}
//等差数列求和模板,[a1,a1+d,...,an]
long long allsum(long long a1,long long an,long long n)
{
if(n%==)
return (a1+an)*(n/);
else return ((a1+an)/)*n;
} int main(int argc, const char * argv[]) {
mobus();
int b,d;
while(scanf("%d%d",&b,&d)!=EOF)
{
int m = min(b,d);
long long ans = ;
while(m>=)
{
int tb = b/( b/m + )+;
int td = d/( d/m + )+;
//前进的最大位置
int tm = max(tb,td);
ans += allsum(tm,m,m-tm+)%MOD*gaobili(b/m, d/m)%MOD;
ans %= MOD;
m = tm-;
}
cout<<(ans+MOD)%MOD<<endl;
}
return ;
}
/*
4 5 */

2.O(n)预处理,每次查询n^0.5

因为bzoj2693题目找不到了,所以直接用了这题来测试。

这题首先是一个经典的公式变形。 交换连加时变量的位置。

而根据第二个重要的性质,乘性函数的乘除之后还是乘性函数。(加减并不是)

所以后面的连加部分也是乘性函数,这时只需要的单独看只含一个因子的时候,因为里面含有u(i),所以对于D=x^k(x是素因子)只有当i = 1 或 x 时不为0,所以

后面的为x^k(1-x)。这时可以在线性筛选时顺便求出来。

**************************************************************
Problem:
User: chenhuan001
Language: C++
Result: Accepted
Time: ms
Memory: kb
****************************************************************/ #include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <algorithm>
#include <iostream>
using namespace std;
//--莫比乌斯反演函数--//
//说明:利用线性素数筛选顺便求了个mu
//注释部分为求从区间[1,b]和区间[1,d]中取两个数,互质对数O(n^0.5)
//复杂度:O(n)
#define N 10000010
bool mark[N];
int prime[N/]; long long sum[N]; #define MOD 20101009 void mobus()
{
int pcnt=; sum[] = ;
for(int i=;i<N;i++)
{
if(mark[i] == )
{
prime[pcnt++] = i; sum[i] = (long long)i*(-i)%MOD;
}
for(int j=;j<pcnt && i*prime[j]<N;j++)
{
int tmp = i*prime[j];
mark[tmp] = ;
if( i%prime[j] == )
{
sum[tmp] = sum[i]*prime[j];
sum[tmp] %= MOD; break;
}
else
{
sum[tmp] = sum[i]*(sum[prime[j]])%MOD;
} }
}
for(int i=;i<N;i++)
sum[i] = (sum[i]+sum[i-])%MOD;
} long long gaobili(int b,int d)
{
if(b<=||d<=) return ;
long long m = min(b,d);
long long ans = ;
while(m>=)
{
long long tb = b/( b/m + )+;
long long td = d/( d/m + )+;
//前进的最大位置
long long tm = max(tb,td);
ans += (sum[m]-sum[tm-])*(((b/m+)*(b/m)/)%MOD)%MOD*(((d/m+)*(d/m)/)%MOD)%MOD;
ans %= MOD;
m = tm-;
}
return (ans+MOD)%MOD;
} int main()
{
mobus();
int b,d;
while(scanf("%d%d",&b,&d)!=EOF)
{
printf("%d\n",(int)gaobili(b,d));
}
return ;
}

至此mobus大概都刷了一遍,原以为很复杂的东西,其实也不是很难。以后面对的题目可能有更多的公式变形,或许难在找出莫比乌斯模型。但基本的也就是这些方法了。

治好了多年的公式恐惧症。。

bzoj2154(莫比乌斯反演)的更多相关文章

  1. 【BZOJ2154】Crash的数字表格(莫比乌斯反演)

    [BZOJ2154]Crash的数字表格(莫比乌斯反演) 题面 BZOJ 简化题意: 给定\(n,m\) 求\[\sum_{i=1}^n\sum_{j=1}^mlcm(i,j)\] 题解 以下的一切都 ...

  2. Bzoj2154 Crash的数字表格 乘法逆元+莫比乌斯反演(TLE)

    题意:求sigma{lcm(i,j)},1<=i<=n,1<=j<=m 不妨令n<=m 首先把lcm(i,j)转成i*j/gcd(i,j) 正解不会...总之最后化出来的 ...

  3. 【莫比乌斯反演】BZOJ2154 Crash的数字表格

    Description 求sigma lcm(x,y),x<=n,y<=m.n,m<=1e7. Solution lcm没有什么直接做的好方法,用lcm=x*y/gcd转成gcd来做 ...

  4. 【bzoj2154】Crash的数字表格 莫比乌斯反演

    题目描述 今天的数学课上,Crash小朋友学习了最小公倍数(Least Common Multiple).对于两个正整数a和b,LCM(a, b)表示能同时被a和b整除的最小正整数.例如,LCM(6, ...

  5. [复习]莫比乌斯反演,杜教筛,min_25筛

    [复习]莫比乌斯反演,杜教筛,min_25筛 莫比乌斯反演 做题的时候的常用形式: \[\begin{aligned}g(n)&=\sum_{n|d}f(d)\\f(n)&=\sum_ ...

  6. 【bzoj2693】jzptab 莫比乌斯反演+线性筛

    题目描述 输入 一个正整数T表示数据组数 接下来T行 每行两个正整数 表示N.M 输出 T行 每行一个整数 表示第i组数据的结果 样例输入 1 4 5 样例输出 122 题解 莫比乌斯反演+线性筛 由 ...

  7. 【BZOJ2693】jzptab [莫比乌斯反演]

    jzptab Time Limit: 10 Sec  Memory Limit: 512 MB[Submit][Status][Discuss] Description   求 Input 第一行一个 ...

  8. 莫比乌斯反演/线性筛/积性函数/杜教筛/min25筛 学习笔记

    最近重新系统地学了下这几个知识点,以前没发现他们的联系,这次总结一下. 莫比乌斯反演入门:https://blog.csdn.net/litble/article/details/72804050 线 ...

  9. hdu1695 GCD(莫比乌斯反演)

    题意:求(1,b)区间和(1,d)区间里面gcd(x, y) = k的数的对数(1<=x<=b , 1<= y <= d). 知识点: 莫比乌斯反演/*12*/ 线性筛求莫比乌 ...

随机推荐

  1. ZOJ 3866 - Cylinder Candy

    3866 - Cylinder Candy Time Limit:2000MS     Memory Limit:65536KB     64bit IO Format:%lld & %llu ...

  2. [小技巧] git: Your branch and 'origin/master' have diverged

    本文参考:http://stackoverflow.com/questions/19864934/git-your-branch-and-origin-master-have-diverged-how ...

  3. SharePoint自动化系列——Site/Web/List级别的导航菜单

    转载请注明出自天外归云的博客园:http://www.cnblogs.com/LanTianYou/ 需求:在不同的测试用例中,对脚本中不确定的因素需要和用户交互来确定,比如选择哪个site,选择哪个 ...

  4. 让VS自动生成我们自己的注释

    1. 找到你VS的安装目录:C:\Program Files (x86)\Microsoft Visual Studio 11.0 2. 在VS安装路径下依次找到这些文件夹:\Common7\IDE\ ...

  5. combotree的加载方法

    <!DOCTYPE html><html><head> <meta http-equiv="Content-Type" content=& ...

  6. linux 开启wifi热点

    1,在网络连接管理中创建一个wifi连接,点击 Add,然后选Wi-Fi 2,设置wifi热点名字.wifi接连名字 3,设置 Mode 选 Ad-hoc,其它默认. 4,在 Wi-Fi Securi ...

  7. Python 2.7.9 Demo - 三元表达式

    #coding=utf-8 #!/usr/bin/python import logging; a = 'abc'; print 'Y' if isinstance(a, str) else 'N';

  8. log4j中文乱码解决方案

    项目中log4j在英文版linux下输出中文日志为乱码. 由于log4j配置文件中没有设置编码格式(encoding),所以log4j就使用系统默认编码.导致乱码. 解决方法是设置编码格式UTF-8, ...

  9. VC++检测当前网络状态

    VC获得本机网络连接状态 转载:http://www.cppblog.com/wrhwww/archive/2010/12/02/135230.html //本机网络连接类型(成功) #define ...

  10. SqlSever基础 print 在消息中输出hello world

    镇场诗:---大梦谁觉,水月中建博客.百千磨难,才知世事无常.---今持佛语,技术无量愿学.愿尽所学,铸一良心博客.------------------------------------------ ...