数值距离(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. ASP.NET Zero--11.一个例子(4)商品分类管理-数据检验

    虽然已经可以添加商品分类,但还需进行优化,比如:用户是否输入.输入字符串是否有格式限制等等. 打开添加分类按钮,名称不输入任何字符,直接保存,会发现列表添加一条空记录.在实际项目中,这是不允许出现的事 ...

  2. 【JS学习笔记】关于function函数

    函数的基本格式 function 函数名() { 代码: } 函数的定义和调用 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transit ...

  3. AsyncTask兼容性

    简介 AsyncTask是Android系统提供的异步方式,其优点在于在子线程执行任务,并将结果传递给主线程. 实现方式 AsyncTask封装了Executor和Handler. 基本使用 通过As ...

  4. iOS 手势识别

    首先给大家解释一下为什么要学习手势识别? 如果想监听一个UIView上面的触摸事件,之前的做法是: 自定义一个UIView : 实现UIView的touches方法,在方法里面实现具体功能 透过tou ...

  5. 创建maven多模块项目(idea工具)

    1.创建父项目模块(new 一个maven空项目模块)不勾选 create from archetype  删除src目录 2.创建子模块 webapp (该模块为web入口模块) 3.创建其他子模块 ...

  6. elasticsearch+spark+hbase 整合

    1.用到的maven依赖 <dependency> <groupId>org.apache.spark</groupId> <artifactId>sp ...

  7. Tiny6410之NAND FLASH驱动

    一.NAND FLASH的特点 S3C6410的NAND FLASH控制器有如下特点 1.自导入模式:复位后,引导代码被送入到8KB的STEPPINGSTONE中,引导代码移动完毕,引导代码将在STE ...

  8. spring+springmvc+mybatis整合框架搭建

    由于例子是基于Maven搭建的,所以首先是pom.xml文件的依赖信息: <project xmlns="http://maven.apache.org/POM/4.0.0" ...

  9. C# lock关键词/lock语句块、线程锁

    一.lock关键词说明 1. lock 关键字将语句块标记为临界区,方法是获取给定对象的互斥锁,执行语句,然后释放该锁. 2. lock 语句块锁定,功能等同于 Monitor.Enter(obj): ...

  10. 杂谈3之English

    1.面向过程(OPP):Orient Procedure Program (C语言) 2.面向对象(OOP):Orient ObjectProgram(Java) 3.面向对象的三大特征:继承Inhe ...