Pride -codeforces 892C

题目

 

You have an array a with length n, you can perform operations. Each operation is like this: choose two adjacent elements from a, say x and y, and replace one of them with gcd(x, y), where gcd denotes the greatest common divisor.

What is the minimum number of operations you need to make all of the elements equal to 1?

Input

The first line of the input contains one integer n (1 ≤ n ≤ 2000) — the number of elements in the array.

The second line contains n space separated integers a1, a2, ..., an (1 ≤ ai ≤ 109) — the elements of the array.

Output

Print -1, if it is impossible to turn all numbers to 1. Otherwise, print the minimum number of operations needed to make all numbers equal to 1.

Examples

Input

5
2 2 3 4 6

Output

5

Input

4
2 4 6 8

Output

-1

Input

3
2 6 9

Output

4

题目大意 :

输入一串数字

然后找相邻两个数字的最大公约数,来替换掉其中一个。通过这样的方法来使这一串数字全部变成1

如果可以那么输出变化次数,如果不可以那么输出-1

算法

 没有什么算法

代码

 

#include<iostream>
using namespace std;
int gcd(int c,int b)                  //递归求最大公约数 
{
	if(b==0)
		return c;
	return gcd(b,c%b);
}                     
int main()
{
	int n,i,j,temp,sub,ans=1000000,sum=0;
	cin>>n;
	int a[n+1];               
	for(i=1;i<=n;i++)               //输入 
	{
		cin>>a[i];
		if(a[i]==1) sum++;      
	}                           
	if(sum!=0)
	{
		cout<<n-sum<<endl;
		return 0;
	}                             //如果这一串数字中含1,那么可以直接输出结果 
	 
	for(i=1;i<=n-1;i++)
	{
		temp=a[i];sub=1000000;
		for(j=i+1;j<=n;j++)
		{
			temp=gcd(temp,a[j]);      //temp每一步都更新 ,每一次求得相邻两个数的最大公约数,就把他赋值给两个中的后一个 
			if(temp==1)  
			{
				sub=j-i;             //一旦最大公约数是一,那么就可以转换为全唯一的数列,那么继续比较操作次数 
				break;
			}
		}
		ans=min(sub+n-1,ans);        //求最小的操作次数 
	}
	if(ans<1000000) cout<<ans<<endl; //如果这个操作次数比1000000小了,说明前面的循环找到了可以改变的方案,并对ans进行了赋值 
	else cout<<"-1"<<endl;

	return 0;
}

解析

首先要清楚,这一串数字中只要出现了 1,那么他肯定可以转换为全为一的序列,操作次数是n-1。(n=数列长度)

所以在输入的时候就计算等于1的元素个数,一但sum不是0,那么就说明存在,直接输出 return 0。

其次,两重循环,找一个区间,使这个区间的头尾最大公约数为1,具体点说是,使这个区间经过 |区间长度|次操作后,出现了数字1(因为一旦出现1就可以直接输出结果)。 

这两重循环中,temp一开始取a[i],之后不断变化为gcd(temp,a[j]),其实就是每一次求得相邻两个数的最大公约数,就把他赋值给两个中的后一个。  这个时候就会有人问 为什么要把最大公约数赋值给后一个,而不是前一个呢 (一会告诉你

一旦求出这个最短出现1的区间,那么结果就是 区间长度+n-1

最大公约数赋值给前一个还是后一个:

举个例子:2,6,9  任何相邻两个数的gcd都不为1,看6这个数的位置,它可以被替换成和2的gcd,再和9求gcd,或被替换成和9的gcd,再和2求gcd,结果是一样的,都是1

一开始我也搞不懂 后来看来别人的博客

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
这是一道 CodeForces 上的题目,题目编号为 749C,题目名称为 Voting。 题目描述: 有 $n$ 个人参加选举,选出一位领导人。每个人都会投票,你知道了每个人选择谁,并且可以知道选票中作废票和弃权票的数量。如果有一个人获得了半数以上的有效选票(即除去作废票和弃权票的票数),那么他将成为领导人。如果没有任何一个人获得半数以上的有效选票,则选举无效。 现在你可以修改任意数量的作废票或弃权票,使选举有效,并使你支持的候选人成为领导人。你需要最少的修改次数。 输入格式: 第一行包含三个整数 $n, a, b$,分别表示选民的数量,作废票的数量和弃权票的数量。 接下来 $n$ 行,每行包含一个字符串,表示每个人的投票情况。如果该字符串为 “YES” 或 “NO”,表示该选民对应的是弃权票;如果该字符串为 “POLL”, 表示该选民对应的是作废票;如果该字符串为其他字符串,则表示该选民对应的是有效选票,该字符串为该选民的投票对象。 输出格式: 如果无法通过修改使选举有效,则输出 -1;否则输出最少的修改次数,使得选举有效,并且你支持的候选人成为领导人。 数据范围: $1 \leq n \leq 2000, 0 \leq a, b \leq n$ 输入样例 #1: ``` 7 2 1 YES NO POLL YES YES YES NO ``` 输出样例 #1: ``` 1 ``` 样例 #1 解释: 总共有7个选民,其中弃权票有3个,作废票有2个,有效选票有2个。由于选民投票意见分散,无法确定一位领导人,因此选举无效。 我们可以将2张作废票修改为支持你所支持的候选人,这样你所支持的候选人将获得3张有效选票,超过半数,成为领导人。因此修改次数为1。 输入样例 #2: ``` 3 1 0 YES NO YES ``` 输出样例 #2: ``` 0 ``` 样例 #2 解释: 总共有3个选民,其中弃权票有1个,作废票有0个,有效选票有2个。你所支持的候选人获得了2张有效选票,超过半数,成为领导人。由于选民投票意见不分散,选举有效,无需修改任何票。因此修改次数为0。 算法1: (模拟) $O(n)$ 首先计算出除作废票和弃权票之外的票数,如果有一人的得票率超过50%,则选举有效,直接输出0。 否则,需要计算出至少需要修改多少张作废票或弃权票,才能使选举有效,并且支持你所支持的候选人成为领导人。 具体来说,我们可以考虑枚举需要修改的作废票和弃权票的数量,假设需要修改 $i$ 张作废票和 $j$ 张弃权票,使得选举有效。那么,你所支持的候选人需要得到至少 $\lceil \frac{n}{2} \rceil$ 张有效选票。我们可以通过统计当前你所支持的候选人得到的有效选票数 $cnt$,以及当前已经修改的作废票和弃权票的数量 $a'$ 和 $b'$,来判断是否存在一组解 $(i, j)$,使得选举有效。 具体来说,如果当前得票数 $cnt + i \ge \lceil \frac{n}{2} \rceil$,那么选举有效,输出 $i+j$ 即可。如果 $(\lceil \frac{n}{2} \rceil - cnt) \le a' + i \le n-b'-j$,那么也存在一组解 $(i,j)$,使得选举有效,输出 $i+j$ 即可。 如果枚举完所有情况,都无法使选举有效,那么输出 -1。 时间复杂度:$O(n^2)$ C++ 代码

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Cherish_lii

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值