CF449C:Jzzhu and Apples
题意简述
给出正整数n,你要把1-n之间的正整数分成尽可能多组,使得每一组两个数的最大公约数大于1;输出能分成最多组的个数,并按任意顺序输出每组的两个数.
很妙的一道题。
首先我们考虑去处理每个质数的倍数,那么这个质数一定是小于等于 n/2 的,不然它在 n 的范围内是不会有倍数的。
那么我们先把 $1~n/2$ 范围内的所有质数筛出来。
枚举质数
然后我们要怎么用质数去处理答案呢?
首先我们从大到小枚举这些质数,然后去枚举它的倍数。
然鹅这样复杂度不会炸么?不会。 $O(\sigma_{i=2}^{n/2}n/i)$ 的复杂度吧,而且还远达不到。
信仰一下应该是过得了的,实际上确实也过了。
处理倍数
然后我们再考虑如果这些倍数该怎么处理。
我们首先看这些倍数有没有被使用过,没有的话就入栈。
最优性理解
那么为什么能用就入栈的解法最优呢?
其实很简单,因为当前枚举那个数可以被当前的质数整除,那么该数与当前枚举质数的倍数去匹配肯定不会产生更劣的结果。
那么如果有数字剩下来呢?
如果说当前枚举质数的未使用过的倍数有奇数个,那么我们可以将第 2 个倍数与最后一个换一下,然后弹出栈,标记为未使用。
为什么这样做能达到最优?
考虑第二个倍数一定是 2 的倍数,那么把他弹出的话就可以和 2 的倍数去匹配,而且我们每次遇到奇数个的情况都是弹 2 的倍数,那么这些多余的数就可以凑一块儿了。
1 //by Judge
2 #include<vector>
3 #include<cstdio>
4 #include<iostream>
5 #define P make_pair
6 using namespace std;
7 const int M=1e5+11;
8 char sr[1<<21],z[20];int C=-1,Z;
9 inline void Ot(){fwrite(sr,1,C+1,stdout),C=-1;}
10 inline void print(int x,char chr='\n'){
11 if(C>1<<20)Ot();if(x<0)sr[++C]=45,x=-x;
12 while(z[++Z]=x%10+48,x/=10);
13 while(sr[++C]=z[Z],--Z);sr[++C]=chr;
14 } int n,cnt,ans,top,prim[M],is[M],vis[M],num[M];
15 pair<int,int> tmp[M];
16 inline void prep(){
17 for(int i=2;i<=n>>1;++i){
18 if(!is[i]) prim[++cnt]=i;
19 for(int j=1;j<=cnt&&i*prim[j]<=n/2;++j){
20 is[i*prim[j]]|=1;
21 if(i%prim[j]==0) break;
22 }
23 }
24 }
25 int main(){
26 scanf("%d",&n),prep();
27 for(int i=cnt;i;--i){ top=0;
28 for(int j=prim[i];j<=n;j+=prim[i])
29 if(!vis[j]) num[++top]=j,vis[j]|=1;
30 if(top&1) swap(num[2],num[top]),vis[num[top]]=0,--top;
31 for(int j=1;j<=top;j+=2) tmp[++ans]=P(num[j],num[j+1]);
32 } print(ans);
33 for(int i=1;i<=ans;++i)
34 print(tmp[i].first,' '),
35 print(tmp[i].second);
36 return Ot(),0;
37 }
CF449C:Jzzhu and Apples的更多相关文章
- CF449C:Jzzhu and Apples——题解
https://vjudge.net/problem/CodeForces-449C 题目大意:1-n编号的苹果两两一对,他们的最大公约数不为1,求这些对的最大匹配. ———————————————— ...
- Codeforces Round #257 (Div. 2) E题:Jzzhu and Apples 模拟
E. Jzzhu and Apples time limit per test 1 second memory limit per test 256 megabytes input standard ...
- Codeforces 450E:Jzzhu and Apples(构造,数学)
E. Jzzhu and Apples time limit per test: 1 seconds memory limit per test: 256 megabytes input: stand ...
- CF449C Jzzhu and Apples (筛素数 数论?
Codeforces Round #257 (Div. 1) C Codeforces Round #257 (Div. 1) E CF450E C. Jzzhu and Apples time li ...
- Codeforces 449.C Jzzhu and Apples
C. Jzzhu and Apples time limit per test 1 second memory limit per test 256 megabytes input standard ...
- CF449 C. Jzzhu and Apples
/* http://codeforces.com/problemset/problem/449/C cf 449 C. Jzzhu and Apples 数论+素数+贪心 */ #include &l ...
- Codeforces 449C Jzzhu and Apples 贪心 (看题解)
Jzzhu and Apples 从大的质因子开始贪心, 如果有偶数个则直接组合, 如果是奇数个留下那个质数的两倍, 其余两两组合. #include<bits/stdc++.h> #de ...
- CF 450E Jzzhu and Apples 数学+模拟
E. Jzzhu and Apples time limit per test 1 second memory limit per test 256 megabytes input standard ...
- CF449C Jzzhu and Apples
嘟嘟嘟 这道题正解是怎么对的其实我也不清楚,总之靠感性理解吧. 首先当然要把1到n / 2的素数都筛出来,因为两两能配对的数一定都是这些素数的倍数.这也就说明对于(n / 2, n]的素数,他们一定不 ...
随机推荐
- js中得计算问题算式结果拼接成字符串怎么解决
如题:经常遇到类似问题 一种:自定义的弱类型 var savNum=0; var num=$("#numU").val();//jsp页面获得得值 savNum=parseInt( ...
- Leetcode 136 Single Number 仅出现一次的数字
原题地址https://leetcode.com/problems/single-number/ 题目描述Given an array of integers, every element appea ...
- Linux下main函数启动过程【程序员自我修养笔记】【自用】
1. 入口函数和程序初始化 1.1 程序从main开始吗? 当程序执行到main函数的第一行时,很多事情都已经完成了: [证1]如下是一段C语言代码: 代码中可以看到,在程序刚刚执行到main的时候, ...
- openstack服务启动之nova-compute
在openstack中,消息的传递一共有两种方式,分别是:服务之间的传递调用Restful api,服务中模块之间的传递使用消息队列.每一个模块在启动是都会启动消息队列,等待队列中的有消息到来.所以模 ...
- node 跨域请求设置
http.createServer((req,res)=>{ //设置允许跨域的域名,*代表允许任意域名跨域 res.setHeader("Access-Control-Allow-O ...
- 细说REST API安全之认证授权
认证授权包含2个方面:(1)访问某个资源时必须携带用户身份信息,如:用户登录时返回用户access_token,访问资源时携带该参数.(2)检查用户是否具备访问当前资源(url或数据)的权限:访问资源 ...
- ArcGIS出图调整
上周为了出一张高分辨率的结合表,大致学了一下出图的过程. (1)打开基础数据的图层属性栏,将某一项属性值显示出来,如下图设置: 注意,字体尽量设置得小一点,否则出图的时候,字太大,会很乱. (2)打开 ...
- Groovy的脚本统一于类的世界
http://groovy-lang.org/structure.html 3.2. Script class A script is always compiled into a class. Th ...
- Jupyter NoteBook使用快捷键
命令模式 (按键 Esc 开启) Enter : 转入编辑模式 Shift-Enter : 运行本单元,选中下个单元 Ctrl-Enter : 运行本单元 Alt-Enter : 运行本单元,在其下插 ...
- 简单linux命令1
1. cat 一次显示整个文件: cat filename 从键盘创建一个文件: cat> filename 将文件合并: cat file1 file2 > filename 2. ...