IEEEXtreme 10.0 - Inti Sets
这是 meelo 原创的 IEEEXtreme极限编程大赛题解
Xtreme 10.0 - Inti Sets
题目来源 第10届IEEE极限编程大赛
https://www.hackerrank.com/contests/ieeextreme-challenges/challenges/inti-sets
In order to motivate his Peruvian students, a teacher includes words in the Quechua language in his math class.
Today, he defined a curious set for a given positive integer N. He called this set, an Inti set, and defined it as the set of all positive integer numbers that have the number 1 as their single common positive divisor with number N.
The math class about Inti sets was amazing. After class, the students try to challenge to teacher. They each ask questions like this: "Could you tell me the sum of all numbers, between A and B (inclusive), that are in the Inti set of N?"
Since the teacher is tired and he's sure that you are the best in class, he wants to know if you can help him.
Input Format
The first line of input contains an integer Q, 1 ≤ Q ≤ 20, representing the number of students. Each of the next Qlines contain three space-separated integers N, A and B, which represent a query.
Constraints
1 ≤ A ≤ B ≤ N ≤ 10^12
Output Format
The output is exactly Q lines, one per student query. For each query you need to find the sum of all numbers between A and B, that are in the Inti set of N, and print the sum modulo 1000000007.
Sample Input
2
12 5 10
5 1 4
Sample Output
12
10
Explanation
In the sample input, Q = 2, so you have to answer two questions:
In the first question N = 12, A = 5 and B = 10. So you have to find the sum of all numbers between 5 and 10, that are in the Inti set of 12.
Inti set ( 12 ) = { 1, 5, 7, 11, 13, ... }
2 and 4 are not in the Inti set (12) because 12 and these numbers are also divisible by 2.
3 and 9 are not in the Inti set (12) because 12 and these numbers are also divisible by 3.
The numbers in the Inti set, which are in the query's range, are 5 and 7, so answer is ( 5 + 7 ) MOD 1000000007 = 12
In the second question, the numbers in the Inti set of 5 between 1 and 4 are: 1, 2, 3, 4; so the answer is ( 1 + 2 + 3 + 4 ) MOD 1000000007 = 10
题目解析
显然直接求和会超时,可以用容斥原理解决。
用sumOver(5, 10, 1)表示区间[5,10]内为1倍数的数
由于12的质因数为2, 3
sum(区间[5, 10]内与12互质的数) = sumOver(5, 10, 1) - sumOver(5, 10, 2) - sumOver(5, 10, 3) + sumOver(5, 10, 6)
可以通过遍历区间[0,2^2)的每一个数来遍历所有因式的组合,
二进制数形式每一位代表是否存在该因数,1代表存在,0代表不存在,
因数的个数为偶数意味着和需要加上,为奇数意味着需要减去
00代表因数为1, 01代表因数为3, 10代表因数为2, 11代表因数为6
注意需要使用取余运算避免溢出。
复杂度分析

如果有c个质因数,那么需要求2^c个数的和
求每一个和需要常数时间O(1)
数N,至多只有一个大于sqrt(N)的质因数,因此质因数的个数不超过log(sqrt(N))+1
总复杂度为O(sqrt(N))
程序
C++
#include <cmath>
#include <cstdio>
#include <vector>
#include <iostream>
#include <algorithm>
using namespace std; #define MAXN 1000000007 // 区间[a,b]内,所有为x倍数数的和
long long sumOver(long long a, long long b, long long x) {
long long aa = (a + x - ) / x; // 上取整
long long bb = b / x; // 下取整 long long sum; // sum会超过long long的表示范围
if( (aa + bb) % == ) {
sum = (((aa + bb) / ) % MAXN) * ((bb - aa + ) % MAXN);
} else {
sum = ((aa + bb) % MAXN) * (((bb - aa + ) / ) % MAXN);
} return ((sum % MAXN) * (x % MAXN)) % MAXN;
} // 求不大于max的所有素数
// 使用筛选法
void getPrimes(vector<long long> &primes, long long max) {
vector<bool> nums(max, );
for(long long i=; i<max; i++) {
if(nums[i] == false) {
primes.push_back(i);
for(int n=*i; n<max; n+=i) {
nums[n] = true;
}
}
}
} // 对数x进行质因数分解
void getFactors(long long x, vector<long long> &factors, vector<long long> &primes) {
int i = ;
while(x > && i < primes.size()) {
if(x % primes[i] == ) {
factors.push_back(primes[i]);
while(x % primes[i] == ) x /= primes[i];
}
i++;
}
// 小于10^12的数最对有一个大于10^6的质因数
if(x > ) {
factors.push_back(x);
}
} int main() {
/* Enter your code here. Read input from STDIN. Print output to STDOUT */
vector<long long> primes;
getPrimes(primes, ); int T;
cin >> T;
for(int t=; t<T; t++) {
long long x, a, b;
cin >> x >> a >> b;
long long result = ;
vector<long long> factors;
getFactors(x, factors, primes); int factorCount = factors.size();
long long binMax = (long long) << factorCount; // 遍历所有的质因数组合
for(long long bin=; bin<binMax; bin++) {
long long factor = ;
int factorC = ;
for(int i=; i<factorCount; i++) {
if( (bin >> i) & ) {
factor *= factors[i];
factorC ++;
}
} if(factorC % == ) {
result = (result + sumOver(a, b, factor) + MAXN) % MAXN;
}
else {
result = (result - sumOver(a, b, factor) + MAXN) % MAXN;
}
} cout << result << endl;
} return ;
}
IEEEXtreme 10.0 - Inti Sets的更多相关文章
- IEEEXtreme 10.0 - Painter's Dilemma
这是 meelo 原创的 IEEEXtreme极限编程比赛题解 Xtreme 10.0 - Painter's Dilemma 题目来源 第10届IEEE极限编程大赛 https://www.hack ...
- IEEEXtreme 10.0 - Ellipse Art
这是 meelo 原创的 IEEEXtreme极限编程大赛题解 Xtreme 10.0 - Ellipse Art 题目来源 第10届IEEE极限编程大赛 https://www.hackerrank ...
- IEEEXtreme 10.0 - Counting Molecules
这是 meelo 原创的 IEEEXtreme极限编程大赛题解 Xtreme 10.0 - Counting Molecules 题目来源 第10届IEEE极限编程大赛 https://www.hac ...
- IEEEXtreme 10.0 - Checkers Challenge
这是 meelo 原创的 IEEEXtreme极限编程大赛题解 Xtreme 10.0 - Checkers Challenge 题目来源 第10届IEEE极限编程大赛 https://www.hac ...
- IEEEXtreme 10.0 - Game of Stones
这是 meelo 原创的 IEEEXtreme极限编程大赛题解 Xtreme 10.0 - Game of Stones 题目来源 第10届IEEE极限编程大赛 https://www.hackerr ...
- IEEEXtreme 10.0 - Playing 20 Questions with an Unreliable Friend
这是 meelo 原创的 IEEEXtreme极限编程大赛题解 Xtreme 10.0 - Playing 20 Questions with an Unreliable Friend 题目来源 第1 ...
- IEEEXtreme 10.0 - Full Adder
这是 meelo 原创的 IEEEXtreme极限编程大赛题解 Xtreme 10.0 - Full Adder 题目来源 第10届IEEE极限编程大赛 https://www.hackerrank. ...
- IEEEXtreme 10.0 - N-Palindromes
这是 meelo 原创的 IEEEXtreme极限编程大赛题解 Xtreme 10.0 - N-Palindromes 题目来源 第10届IEEE极限编程大赛 https://www.hackerra ...
- IEEEXtreme 10.0 - Mysterious Maze
这是 meelo 原创的 IEEEXtreme极限编程大赛题解 Xtreme 10.0 - Mysterious Maze 题目来源 第10届IEEE极限编程大赛 https://www.hacker ...
随机推荐
- array_uintersect、array_uintersect_assoc、array_uintersect_uassoc 的使用方法
和 array_intersect 类似,只不过 array_uintersect* 系列函数的值比较使用自定义函数: 键的比较,array_uintersect.array_uintersect_a ...
- cpu的核数
相信大多数的人都知道CPU区分单核.双核.四核.六核.八核等,一些电脑小白肯定认为核心越多肯定性能越强,但是不少装机用户发现,有的CPU型号虽然是双核,但是要比一些四核还要贵,很多人感到迷惑,那么 ...
- poj 3261 后缀数组 可重叠的 k 次最长重复子串
Milk Patterns Time Limit: 5000MS Memory Limit: 65536K Total Submissions: 16430 Accepted: 7252 Ca ...
- 替换换行符:回车换行CR/LF
windows采用回车+换行CR/LF表示下一行,UNIX/Linux使用换行符LF表示下一行,MAC OS系统使用用回车符CR表示下一行. CR使用符号'\r'表示, ASCII码是13: LF使用 ...
- 使用git上传项目到GitHub上
之前的博客有<使用git拉取GitHub上的项目>的文章,那么现在说一下,如何上传项目到GitHub上. 1. Git的.gitignore 文档配置 因为项目中可能有很多的图片还有nod ...
- CF832 D LCA倍增 裸
有询问$a,b,c$,求a到c路径上,同时是a到b路径的点的个数.其中询问中的a,b,c可任意选择作为起点或终点,求一组询问中最大值. LCA用于计算树上点对间距离,对于一组询问求深度最大的点作为起点 ...
- MYSQL5.6学习——mysqldump备份与恢复
MYSQL备份 冷备份:停止服务进行备份,即停止数据库的写入 热备份:不停止服务进行备份(在线) l mysql的MyIsam引擎只支持冷备份,InnoDB支持热备份,原因: InnoDB引擎是事务 ...
- 网站开发中很有用的几个 jQuery 地图插件
下面提到的 jQuery 地图插件不仅仅是提供一个简便的方式来安装一个地图,如果你想在它们之间选择一个放到你的网站上,那么它们还有更多的额外选项来提供更多更全面的功能.大部分的 jQuery 地图插件 ...
- Lua的各种资源2
Lua Directory This page is a top level directory of all Lua content at this wiki, grouped by top ...
- 【CodeForces】915 F. Imbalance Value of a Tree 并查集
[题目]F. Imbalance Value of a Tree [题意]给定n个点的带点权树,求所有路径极差的和.n,ai<=10^6 [算法]并查集 [题解]先计算最大值的和,按点权从小到大 ...