bjfu1109 最小公倍数和
这题真是过了n年才a。最早是在2010年北大培训比赛上看到的这题,当时我不会,竹教主也不会,但他记下来了,研究一段时间后就会了,还把这题加到我校oj上。过了这么多年,我上网搜,关于这个问题的解题报告还是没有,于是我花了几天时间做了出来,发布此解题报告。
题目是要求从1到n的所有数与n的最小公倍数的和,再除了n。第一眼看到数据范围,就知道是不能硬做(即从1到n依次算一遍)的。那么,为了找规律,得把公式化一化。
f(n) = [lcm(1,n)+lcm(2,n)+……+lcm(n,n)]/n
= 1/gcd(1,n)+2/gcd(2,n)+……+n/gcd(n,n)
于是f(n)化成了一堆数的和。
显然,每一个gcd(k,n)都是能整除n的,按gcd(k,n)的值进行分类合并,就可以把f(n)整理成以下形式
f(n) = (……)/1 + (……)/2 +……+ (……)/d
这里d就是所有能整除n的整数。而(……)/1的分子为所有与n互质的数的和;(……)/2的分子为所有与n的最大公约数为2,也就是与n/2互质的数的和;依次类推,得到公式

有了这个公式,虽然是前进了一大步,但依然没法解题。因为这还是必须枚举n的所有约数,而这个复杂度依然为O(n)。
接下来就是比较核心的一步,构造。

f(n) = (g(n) + 1) / 2,求得g(n)就立即可得f(n)了。
这里构造g(n)的原因,是因为g(n)有积性性质。
即
对于任意gcd(m, n) = 1,有g(m * n) = g(m) * g(n)。证明略掉,读者自证。
有了积性性质就好办了,将n质因数分解成pi^ci相乘的形式,n = ∏(pi^ci),则g(n) = g(∏(pi^ci)) = ∏(g(pi^ci))
g(pi^ci)很好计算。这里的d就是1, pi, pi^2, pi^3, ..., pi^ci,而phi(pi^ci)=(pi-1)*pi^(ci-1),合起来就是个等比数列,可以推出来一个公式。也就是说可以O(1)时间算出g(pi^ci)。
如此一来,整个问题就可解了。
不过这题还是很变态的,因为数据就有50000组,仅分解质因数写得不好,都可能挂掉。最后我还是过了,代码如下:
/*
* bjfu1109
* Author : ben
*/
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <ctime>
#include <iostream>
#include <algorithm>
#include <queue>
#include <set>
#include <map>
#include <stack>
#include <string>
#include <vector>
#include <deque>
#include <list>
#include <functional>
#include <numeric>
#include <cctype>
using namespace std;
#ifdef ON_LOCAL_DEBUG
#else
#endif
typedef long long LL;
typedef int typec; LL getPow(int a, int b) {
LL res, temp;
res = , temp = (LL) a;
while (b) {
if (b & ) {
res = res * temp;
}
b >>= ;
temp = temp * temp;
}
return res;
} int get_int() {
int res = , ch;
while (!((ch = getchar()) >= '' && ch <= '')) {
if (ch == EOF)
return -;
}
res = ch - '';
while ((ch = getchar()) >= '' && ch <= '')
res = res * + (ch - '');
return res;
} const int N = ;
bool isPrime[N + ];//多用两个元素以免判断边界
vector<int> pt;
void init_prime_table() {
memset(isPrime, true, sizeof(isPrime));
int p = , q, del;
double temp;
while (p <= N) {
while (!isPrime[p]) { p++; }
if (p > N) {//已经结束
break; }
temp = (double) p;
temp *= p;
if (temp > N)
break;
while (temp <= N) {
del = (int) temp; isPrime[del] = false;
temp *= p; }
q = p + ;
while (q < N) {
while (!isPrime[q]) { q++; }
if (q >= N) { break;}
temp = (double) p;
temp *= q;
if (temp > N) break;
while (temp <= N) {
del = (int) temp;
isPrime[del] = false;
temp *= p;
}
q++;
}
p++;
}
for (int i = ; i <= N; i++) {
if (isPrime[i]) {
pt.push_back(i);
}
}
} /**
* p为素数表,至少应该包括sqrt(N)以内的所有素数,f保存结果
* f[i].first表示N的一个素因数,f[i].second为这个素因子的个数
* typec可以是int、long、long long等
*/
typedef vector<pair<typec, int> > FactorList;
void get_prime_factor(const typec &N_, FactorList &f, const vector<int> &p) {
int i, t, n, pl = p.size();
typec N = N_;
f.clear();
for(i = ; i < pl; i++) {
t = p[i];
if (t * t > N_) {
break;
}
if(N % t == ) {
n = ;
while(N % t == ) {
n++; N /= t;
}
f.push_back(make_pair(t, n));
}
if(N == ) { break; }
}
if(N > ) {
f.push_back(make_pair(N, ));
}
} inline LL g(LL p, int c) {
LL ret = getPow(p, * c) - ;
ret = ret / (p + ) * p + ;
return ret;
} FactorList fl;
LL g(int n) {
get_prime_factor(n, fl, pt);
int len = fl.size();
LL ret = ;
for (int i = ; i < len; i++) {
ret *= g(fl[i].first, fl[i].second);
}
return ret;
} int main() {
#ifdef ON_LOCAL_DEBUG
freopen("data.in", "r", stdin);
#endif
LL ans, N;
init_prime_table();
while ((N = get_int()) > ) {
ans = g(N) + ;
ans /= ;
printf("%I64d\n", ans);
}
return ;
}
bjfu1109 最小公倍数和的更多相关文章
- 求N个数的最大公约数和最小公倍数(转)
除了分解质因数,还有另一种适用于求几个较小数的最大公约数.最小公倍数的方法 下面是数学证明及算法实现 令[a1,a2,..,an] 表示a1,a2,..,an的最小公倍数,(a1,a2,..,an)表 ...
- C语言 · 最小公倍数
问题描述 编写一函数lcm,求两个正整数的最小公倍数. 样例输入 一个满足题目要求的输入范例.例:3 5 样例输出 与上面的样例输入对应的输出.例: 数据规模和约定 输入数据中每一个数的范围. 例:两 ...
- Java程序设计之最大公约数和最小公倍数
题目:输入两个正整数number1和number2,求其最大公约数和最小公倍数. 算法:较大数和较小数取余,较小数除余数,一直到余数为0时,为最大公约数(辗转相除法):最大公倍数numbe1*numb ...
- 最大公约数和最小公倍数--java实现
代码: //最大公约数 public int gcd(int p,int q){ if(q == 0) return p; return gcd(q, p % q); } //最小公倍数 public ...
- python 最小公倍数
最小公倍数 求解两个整数(不能是负数)的最小公倍数 方法一:穷举法 def LCM(m, n): if m*n == 0: return 0 if m > n: lcm = m else: lc ...
- 输入两个正整数m和n,求其最大公约数和最小公倍数
public static void main(String[] args){ Scanner sc = new Scanner (System.in); int a,b; System.out ...
- Java编写最大公约数和最小公倍数
package javaapplication24; class NegativeIntegerException extends Exception{ String message; public ...
- poj 3101Astronomy(圆周追击+分数最小公倍数)
/* 本题属于圆周追击问题: 假设已知两个圆周运动的物体的周期分别是a ,b, 设每隔时间t就会在同一条直线上 在同一条直线上的条件是 角度之差为 PI ! 那么就有方程 (2PI/a - 2PI/b ...
- 【codevs1012】最大公约数和最小公倍数
题目描述 Description 输入二个正整数x0,y0(2<=x0<100000,2<=y0<=1000000),求出满足下列条件的P,Q的个数 条件: 1.P,Q是正整 ...
随机推荐
- leetcode:两个数的和||
两个数的和|| 给定一个排序数组,求出其中两个数的和等于指定target时,这两个数在原始数组中的下标,返回的下标从1开始 解题 原始数组已经是升序的,找出其中两个数的和等于target 定义两个指针 ...
- Unity UGUI —— 鼠标穿透UI问题(Unity官方的解决方法)
解决方案 : http://www.cnblogs.com/fly-100/p/4570366.html 这里我们直接在使用Input.GetMouseButtonDown(0)的地方加了一个检测函数 ...
- 【nginx网站性能优化篇(3)】反向代理实现负载均衡
注意,本篇文章为负载均衡的理论篇,后续找个机会推出实战篇.理论篇主要讲述如何配置负载均衡,配置负载均衡不难.难的是真正的实战,比如如何做到多服务器之间的数据共享(session,file等),多cac ...
- SQL Server数据导入导出的几种方法
在涉及到SQL Server编程或是管理时一定会用到数据的导入与导出, 导入导出的方法有多种,结合我在做项目时的经历做一下汇总: 1. SQL Server导入导出向导,这种方式是最方便的. 导入向导 ...
- Loongnix 系统(MIPS Linux)
电脑上的x86,手机上的ARM,在各自领域都是很成熟的CPU架构了,龙芯也参与进去竞争是很难的,就算是Intel,挤破头皮疯狂补贴自家的Atom x86还是在手机领域无法立足. 所以说,个人觉得龙芯可 ...
- 不重启使XP环境变量生效
不重启使XP环境变量生效 http://www.pkghost.cn/wz/sort0185/8874.html 在“我的电脑”->“属性”->“高级”->“环境变量”中增加或修改环 ...
- oracle 修改表的sql语句
oracle 修改表的sql语句 1增加一个列:ALTER TABLE 表名 ADD(列名 数据类型);如:ALTER TABLE emp ADD(license varchar2(256)) ...
- javascript精确计算
一篇文章: 4 个用于执行高级数学计算的 JavaScript 库 numbers.js Numeric Javascript accounting.js Tangle 有时只需要加减乘法能精确,没 ...
- Android开发之创建App Widget和更新Widget内容
App WidgetsApp Widgets are miniature application views that can be embedded in other applications (s ...
- openfire中mysql的前期设置
使用openfire的时候如果需要使用自己的mysql数据库,需要提前进行设置,下面将记录下,基本的设置过程. 一.前期准备工作: 1.先下载两个工具一个是mysql数据库还有一个是SQLyog(可以 ...