数值距离(dis.pas/c/cpp)
题目大意
我们可以对一个数 x 进行两种操作:
1、 选择一个质数 y,将 x 变为 x*y
2、 选择一个 x 的质因数 y,将 x 变为 x/y
定义两个数 a,b 之间的距离为把 a 变成 b 所需要执行的最少操作次数。例如数 69 与 42
之间的距离为 3,因为 42=69/23*2*7
现在有一个长度为 n 的序列 a1,a2,…,an。对于每一个 i,我们需要找到一个 j,使得 ai 到
aj 的距离最小,若有多个 j 满足条件,输出最小的 j。
输入文件
输入文件为 dis.in。
输入共有 n+1 行,第一行有一个数 n,接下来 n 行每行一个数 ai。
输出文件
输出文件为 dis.out。
输出一共 n 行,每行一个整数,第 i 行的整数表示对于 i 所求得的 j 是多少。
样例输入
6
1
2
3
4
5
6
样例输出
2
1
1
2
1
2
数据规模与约定
对于 30%的数据,n≤1000;
另有 20%的数据,ai≤1000;
对于 100%的数据,2≤n≤100000,1≤ai≤1000000。

——————————————————题解

这道题的暴力思路就是枚举每两个数,然后算这两个数的gcd,然后两个数的分解质因数个数的和减去2*gcd分解质因数的个数就是距离,找最小,这是(n^2logn)的

大概12,16这两个数,gcd是4,4=2*2,12=2*2*3,16=2*2*2*2,12和16的距离就是(4+3)-2*2

这样就可以轻松愉快的拿到30分了!【然而我很弱,爆了0……orz】

然后我们发现每两个数的距离(不一定是最小距离)可以写作这两个数分解质因数个数和减去他们俩共同的约数的分解质因数个数

我们可选择不枚举数,而是枚举他们的约数,显然约数为gcd的时候距离这小,但这会在不断更新约数的时候被更新到。

设约数为k,然后我们在a1,a2...an的序列里找到k的倍数并向k连一条边,由于a约数的对数不会超过a^0.5,所以我们枚举的次数不会超过10^3n (10^6^0.5)

然后我们暴力的枚举到一个约数离它最近的最小数和次小数,次小数更新最小数(因为最小数不能自我更新),最小数更新其他数。也是10^3n的。

这道题输出的是下标!!!这道题输出的是下标!!!这道题输出的是下标!!!这道题输出的是下标!!!orz自己真是越来越瞎了

还有要学会用pair……

【注意:这里的ans并非输出的答案,而是最小的距离】

 #include <iostream>
#include <cstdio>
#include <algorithm>
#include <vector>
#include <queue>
#include <cmath>
#define siji(i,x,y) for(int i=(x);i<=(y);i++)
#define gongzi(j,x,y) for(int j=(x);j>=(y);j--)
#define xiaosiji(i,x,y) for(int i=(x);i<(y);i++)
#define sigongzi(j,x,y) for(int j=(x);j>(y);j--)
#define pii pair<int,int>
#define fi first
#define se second
using namespace std;
vector<pii> list[];
int n,num[];
pii ans[];
int factor[],prime[],cnt;
bool isprime[];
void init() {
scanf("%d",&n);
siji(i,,) {
if(!isprime[i]) {factor[i]=;prime[++cnt]=i;}
for(int j=;prime[j]*i<=;j++) {
isprime[prime[j]*i]=;
factor[prime[j]*i]=factor[prime[j]]+factor[i];
//质因子是什么不重要,我们只要知道个数就可以了
if(i%prime[j]==) break;
}
}
siji(i,,n) {
scanf("%d",&num[i]);
ans[i].fi=0x1f1f1f1f;
int t=(int)sqrt(num[i]);//从1-a[i]^0.5枚举看看那些是它的约数,是它的约数就有可能是它和别的数的gcd
siji(j,,t) {
if(num[i]%j== && j*j!=num[i]) {
list[j].push_back(pii(factor[num[i]/j],i));
list[num[i]/j].push_back(pii(factor[j],i));
}
else if(num[i]%j== && j*j==num[i]) {
list[j].push_back(pii(factor[j],i));
}
}
} } void solve() {
siji(i,,) {
if(list[i].size()>) {
pii t1=list[i][],t2=list[i][];//讲真pair会省掉很多比较的麻烦,orz蒟蒻的自言自语罢了
//在比较时first的权重较大,所以first小的在前面,first相同再是second,second小的在前面,perfect
if(t2<t1) swap(t1,t2);
int s=list[i].size();
xiaosiji(j,,s) {
if(t2>list[i][j]) t2=list[i][j];
if(t2<t1) swap(t1,t2);
}
xiaosiji(j,,s) {
if(list[i][j]!=t1) {
ans[list[i][j].se]
=min(ans[list[i][j].se],pii(t1.fi+list[i][j].fi,t1.se));
}
}
ans[t1.se]=min(ans[t1.se],pii(t1.fi+t2.fi,t2.se)); }
}
siji(i,,n) {
printf("%d\n",ans[i].se);
}
}
int main(int argc, char const *argv[])
{
freopen("dis.in","r",stdin);
freopen("dis.out","w",stdout);
init();
solve();
return ;
}

辽宁OI2016夏令营模拟T1-dis的更多相关文章

  1. 辽宁OI2016夏令营模拟T2-road

    最短路(road.pas/c/cpp)题目大意有一个点数为 n,边数为 m 的无向图,点的编号为 1 到 n.边的权值均为非负数.现在请你求出从点 1 到点 n 的最短路径条数,若有无限条则输出-1, ...

  2. 辽宁OI2016夏令营模拟T3-chess

    放棋子(chess.pas/c/cpp)题目大意现在有一个 n*m 的棋盘,现在你需要在棋盘上摆放 2n 个棋子,要求满足如下条件:1. 每一列只能有一个棋子:2. 每一行的前 xi 个格子有一个棋子 ...

  3. 20181229模拟 T1 palindrome

    20181229模拟 T1 palindrome 题意 : \(S\)是字符串\(s\)的子串可重集,求\(\sum\limits_{x\in S}\sum\limits_{y\in S}(|x|+| ...

  4. 模拟T1数字number

    那么第一题首先非常水的一道题…… 看一下题 数字(number) Time Limit:1000ms   Memory Limit:128MB 题目描述 LYK拥有n个数,这n个数分别是a1,a2,… ...

  5. 洛谷金秋夏令营模拟赛 第2场 T11737 时之终末

    这道题就是道状压dp...比赛的时候太贪心 然后状压又不好 所以T2 T3一起挂了QAQ 吸取教训QAQ f[i][j][k]表示前i个数选了j个 最后a个的状态为k的答案 #include<c ...

  6. 洛谷金秋夏令营模拟赛 第2场 T11738 伪神

    调了一个下午只有八十分QAQ md弃了不管了 对拍也没拍出来 鬼知道是什么数据把我卡了QAQ 没事我只是个SB而已 这题其实还是蛮正常的 做法其实很简单 根据链剖的构造方法 你每次修改都是一段又一段的 ...

  7. 「题解」:07.16NOIP模拟T1:礼物

    问题 A: 礼物 时间限制: 1 Sec  内存限制: 256 MB 题面 题目描述 夏川的生日就要到了.作为夏川形式上的男朋友,季堂打算给夏川买一些生 日礼物. 商店里一共有种礼物.夏川每得到一种礼 ...

  8. 51nod1185 威佐夫游戏 V2 (模拟乘法)

    1185 威佐夫游戏 V2  基准时间限制:1 秒 空间限制:131072 KB 分值: 0 难度:基础题  收藏  关注 有2堆石子.A B两个人轮流拿,A先拿.每次可以从一堆中取任意个或从2堆中取 ...

  9. LUOGU NOIP 2018 模拟赛 DAY1

    T1 传送门 解题思路 这似乎是小学数学知识???mod 9就相当于各位之和mod 9,打表求了个逆元,等差数列求和公式就行了. #include<iostream> #include&l ...

随机推荐

  1. openresty 前端开发轻量级MVC框架封装一(控制器篇)

    通过前面几章,我们已经掌握了一些基本的开发知识,但是代码结构比较简单,缺乏统一的标准,模块化,也缺乏统一的异常处理,这一章我们主要来学习如何封装一个轻量级的MVC框架,规范以及简化开发,并且提供类似p ...

  2. 驱动04.平台总线驱动模型——点亮LED灯

    1 平台总线的简介 平台总线是一种虚拟的总线,相应的设备则为platform_device,而驱动则为platform_driver.总线将设备和驱动绑定,在系统每注册一个设备的时候,会寻找与之匹配的 ...

  3. USACO 4.2 Drainage Ditches(网络流模板题)

    Drainage DitchesHal Burch Every time it rains on Farmer John's fields, a pond forms over Bessie's fa ...

  4. python绝技 — 搜寻蓝牙设备

    需要安装蓝牙模块:pybluez sudo pip install pybluez 代码 #!/usr/bin/env python #--*--coding=utf-8--*-- #P191 #su ...

  5. 使用Gradle编译release apk报错:Please correct the above warnings first

    在开发SDK的过程中,遇到了一个研发,使用了自己的SDK之后遇到了各种问题,于是我们自己帮忙接入. 所有代码都接入完成之后,准备export出一个release包,但是此时却报错: 此时出现了很多的w ...

  6. Ubuntu16.04删除客人会话

    1.按下 Ctrl+Alt+T - 打开终端 2.输入以下指令: sudo gedit /etc/lightdm/lightdm.conf 3.源代码之后添加如下代码,然后保存.关闭,重启电脑即可. ...

  7. Tensorflow (1)

    'tf.placeholder' or 'tf.Variable' The difference is that with tf.Variable you have to provide an ini ...

  8. sqlcipher for android

    github 地址 https://github.com/sqlcipher/android-database-sqlcipher 官网教程 https://www.zetetic.net/sqlci ...

  9. redis配置文件redis.conf的参数说明

    打开redis.conf文件: # By default Redis does not run as a daemon. Use 'yes' if you need it. # Note that R ...

  10. CodeForces 708B Recover the String

    构造. 根据$a[0][0]$可以求得$0$的个数$p$,根据$a[1][1]$可以求得$1$的个数$q$. 如果找不到$p$或$q$,那么就无解. 每一个$0$放到序列中的任何一个位置,假设和前面的 ...