Educational Codeforces Round 37-G.List Of Integers题解
一、题目

二、题目链接
http://codeforces.com/contest/920/problem/G
三、题意
给定一个$t$,表示有t次查询。每次查询给定一个$x$, $p$, $k$,需要输出一个大于$x$、与$p$互质、第$k$大的数字。如样例1所示,比$7$大、与$22$互质、第$1$大的数字是$9$,第$2$大的数字是$13$,第$3$大的数字是$15$。
四、思路
二分+容斥原理。
二分一个数字$mid$,看看$[x+1,mid]$之间与$p$互质的数的个数。需要注意的是,如果个数是$k$,还要判断二分的$mid$是不是与$p$互质。如果是,说明要输出的值在$[x+1, mid)$之间,要需要继续做二分。(注意括号样式哦)
统计$[x+1, mid]$之间与$p$互质的数的个数,方法是:把这个区间内$p$的素因子的倍数全部筛掉,剩余的就是与$p$互质的了。在筛的过程中,比如$p$是$12$,素因子是$2$、$3$,筛掉$2$的倍数的方法是:区间长度$len - (mid / 2 - x / 2)$。同理,筛掉$3$的倍数的方法也一样。但是,对于所有$6$的倍数,会被减两次,所以,需要再加一次。而这个过程,其实就是容斥原理。
在这个题目中,利用容斥原理统计个数时,有两个写法:
1、每次二分一个上界值$mid$,就枚举二进制;
2、保存所有的询问,对每个询问的$p$,都预处理出它的所有素因子组合的乘积。然后,再枚举每一个询问,传入二分上界值$mid$时,枚举预处理的$p$的所有素因子组 合的乘积即可。(详细的写法可参考源代码)
另外,还有一点,在做询问之前,需要预处理出$[1, 1e6]$区间内所有数字的素因子。
五、源代码
1、写法一:每次二分一个上界值$mid$,就枚举二进制;
#pragma GCC optimize(2)
#pragma comment(linker, "/STACK:102400000, 102400000")
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
;
template <class T> inline void read(T &x) {
int t;
bool flag = false;
')) ;
if(t == '-') flag = true, t = getchar();
x = t - ';
+ t - ';
if(flag) x = -x;
}
bool prime[MAXN];
vector<LL> d[MAXN];//d[i]:数字i的所有素因子
void init() {
;
; i <= N; ++i)d[i].clear(), prime[i] = true;
; i <= N; ++i) {
if(prime[i]) {
for(int j = i + i; j <= N; j += i)prime[j] = false;
}
}
; i <= N; ++i) {
if(prime[i]) {
for(int j = i; j <= N; j += i)d[j].push_back(LL(i));
}
}
}
/**[x+1, mid]**/
LL calc(LL x, LL p, LL mid) {
LL res = mid - x;
/*枚举二进制:
如果i=5,二进制为0101,从低位数起,第0位和第2位为1,
那么就取d[p]的第0个和第2个素因子,计算乘积,去做容斥操作。
同时,需要统计取得素因子的个数。
如果是奇数,那么,容斥操作中符号是+,否则是-。
*/
, t = << d[p].size(); i < t; ++i) {
LL cnt = , prod = ;
; j > ; j >>= , ++k) {
) {
cnt++;
prod *= d[p][k];
}
}
res -= (mid / prod - x / prod) * (cnt % == ? : -);
}
return res;
}
int main() {
#ifndef ONLINE_JUDGE
freopen("Ginput.txt", "r", stdin);
#endif // ONLINE_JUDGE
LL T, x, p, k, low, high, mid, ans = , tmp;
init();
read(T);
while(T--) {
read(x), read(p), read(k);
low = x, high = 1LL << ;/*这个地方要特别注意,写大了就超时。*/
) {
mid = (low + high) / ;
tmp = calc(x, p, mid);
if(tmp < k)low = mid;
else if(tmp > k)high = mid;
else {
if(__gcd(mid, p) > 1LL)high = mid;
else {
ans = mid;
break;
}
}
}
printf("%lld\n", ans);
}
;
}
2、保存所有的询问,对每个询问的$p$,都预处理出它的所有素因子组合的乘积。然后,再枚举每一个询问,传入二分上界值$mid$时,枚举预处理的$p$的所有素因子组 合的乘积;
#pragma GCC optimize(2)
#pragma comment(linker, "/STACK:102400000, 102400000")
#include<bits/stdc++.h>
using namespace std;
typedef long long LL;
typedef tuple<LL, LL, LL> T3L;
typedef pair<LL, LL> P2L;
;
template <class T> inline void read(T &x) {
int t;
bool flag = false;
')) ;
if(t == '-') flag = true, t = getchar();
x = t - ';
+ t - ';
if(flag) x = -x;
}
bool prime[MAXN];
vector<LL> d[MAXN];/*d[i]:数字i的素因子*/
vector<P2L> f[MAXN];/*f[i]:数字i的素因子组合的乘积和该组合的素因子个数*/
vector<T3L> qs;
void init() {
;
qs.clear();
; i <= N; ++i)d[i].clear(), f[i].clear(), prime[i] = true;
; i <= N; ++i) {
if(prime[i]) {
for(int j = i + i; j <= N; j += i)prime[j] = false;
}
}
; i <= N; ++i) {
if(prime[i]) {
for(int j = i; j <= N; j += i)d[j].push_back(LL(i));
}
}
}
/**[x+1, mid]**/
LL calc(LL x, LL p, LL mid) {
LL res = mid - x;
; i < f[p].size(); ++i) {/*直接枚举素因子组合的乘积*/
LL prod = f[p][i].first, cnt = f[p][i].second;
res -= (mid / prod - x / prod) * (cnt % == ? : -);
}
return res;
}
int main() {
#ifndef ONLINE_JUDGE
freopen("Ginput.txt", "r", stdin);
#endif // ONLINE_JUDGE
LL T, x, p, k, low, high, mid, ans = , tmp;
init();
read(T);
; i < T; ++i) {
read(x), read(p), read(k);
qs.push_back(make_tuple(x, p, k));/*保存所有询问*/
}
/*预处理每个询问的p的素因子组合的乘积、组合个数。
写法同方法1。
*/
; t < T; ++t) {
tmp = >(qs[t]);
) {
, top = << d[tmp].size(); i < top; ++i) {
LL cnt = , prod = ;
; j > ; j >>= , ++k) {
) {
cnt++;
prod *= d[tmp][k];
}
}
f[tmp].push_back(make_pair(prod, cnt));
}
}
}
; t < T; ++t) {
x = >(qs[t]), p = >(qs[t]), k = >(qs[t]);
low = x, high = 1LL << ;/*这个地方要特别注意,写大了就超时。*/
) {
mid = (low + high) / ;
tmp = calc(x, p, mid);
if(tmp < k)low = mid;
else if(tmp > k)high = mid;
else {
if(__gcd(mid, p) > 1LL)high = mid;
else {
ans = mid;
break;
}
}
}
printf("%lld\n", ans);
}
;
}
Educational Codeforces Round 37-G.List Of Integers题解的更多相关文章
- Educational Codeforces Round 37 G. List Of Integers (二分,容斥定律,数论)
G. List Of Integers time limit per test 5 seconds memory limit per test 256 megabytes input standard ...
- Educational Codeforces Round 37
Educational Codeforces Round 37 这场有点炸,题目比较水,但只做了3题QAQ.还是实力不够啊! 写下题解算了--(写的比较粗糙,细节或者bug可以私聊2333) A. W ...
- Educational Codeforces Round 37 (Rated for Div. 2)C. Swap Adjacent Elements (思维,前缀和)
Educational Codeforces Round 37 (Rated for Div. 2)C. Swap Adjacent Elements time limit per test 1 se ...
- Educational Codeforces Round 63 (Rated for Div. 2) 题解
Educational Codeforces Round 63 (Rated for Div. 2)题解 题目链接 A. Reverse a Substring 给出一个字符串,现在可以对这个字符串进 ...
- Educational Codeforces Round 65 (Rated for Div. 2)题解
Educational Codeforces Round 65 (Rated for Div. 2)题解 题目链接 A. Telephone Number 水题,代码如下: Code #include ...
- Educational Codeforces Round 64 (Rated for Div. 2)题解
Educational Codeforces Round 64 (Rated for Div. 2)题解 题目链接 A. Inscribed Figures 水题,但是坑了很多人.需要注意以下就是正方 ...
- Educational Codeforces Round 58 (Rated for Div. 2) 题解
Educational Codeforces Round 58 (Rated for Div. 2) 题目总链接:https://codeforces.com/contest/1101 A. Min ...
- Educational Codeforces Round 60 (Rated for Div. 2) 题解
Educational Codeforces Round 60 (Rated for Div. 2) 题目链接:https://codeforces.com/contest/1117 A. Best ...
- Educational Codeforces Round 37 (Rated for Div. 2) G
G. List Of Integers time limit per test 5 seconds memory limit per test 256 megabytes input standard ...
- codeforces 920 EFG 题解合集 ( Educational Codeforces Round 37 )
E. Connected Components? time limit per test 2 seconds memory limit per test 256 megabytes input sta ...
随机推荐
- CSS 再学习,基础篇
语法 h1 {color:red; font-size:14px;} 共享声明 h1,h2,h3,h4,h5,h6 { color: green; } 继承 通过 CSS 继承,子元素将继承最高级元素 ...
- 使用由 Intel MKL 支持的 R
我们通常使用的 R 版本是单线程的,即只使用一个 CPU 线程运行所有 R 代码.这样的好处是运行模型比较简单且安全,但是它并没有利用多核计算.Microsoft R Open(MRO,https:/ ...
- git关于 LF 与 CRLF
问题描述: 今天在idea上配置好了git操作到push,目测都没遇到什么问题. 但是再次点击提交按钮的时候,仍然提示我所有的文件都是需要提交的(我的主题中蓝色显示) 然而本地并无改动,在提交窗口得到 ...
- C# winform实现右下角弹出窗口结果的方法
using System.Runtime.InteropServices; [DllImport("user32")] private static extern bool Ani ...
- 获取文本中所有的<img>标签的位置,获取所有img标签的src
public static int[] GetImagePos(string str) { str = str.Replace("$", " "); str = ...
- Dubbo原理简介、与Zookeeper整合利用
官方文档:http://dubbo.io/books/dubbo-user-book/ Dubbo的简单介绍 Dubbo是一个分布式服务框架,架构如图: 节点角色说明: Provider: 暴露服务的 ...
- Linux 的 Out-of-Memory (OOM) Killer
同事在 Linux 服务器上遇到点小问题,我也上去折腾半天.这还是第一次注意到 Linux 这个多年来就存在的特性:OOM Killer .说白了 OOM Killer 就是一层保护机制,用于避免 L ...
- 通过java解析域名获得IP地址
IP地址是Internet主机的作为路由寻址用的数字型标识,人不容易记忆.因而产生了域名(domain name)这一种字符型标识. DNS即为域名解析服务.在这里我们如果想通过java程序来解析域名 ...
- [转载]队列queue和双端Dequeue
转载自:http://uule.iteye.com/blog/2095650?utm_source=tuicool 注意:这都只是接口而已 1.Queue API 在java5中新增加了java.ut ...
- LNMP架构下Discuz论坛的搭建
在上一节中,我们对lnmp架构下的mysql.php.nginx进行源码的安装,并设置了相关的安装参数.现在我们将在上一节的基础上,把三者联系起来进行一个论坛的部署. 一.首先进行Discuz(社区论 ...