[蓝桥杯] 最大比例

峰值内存消耗 < 256M  CPU消耗  < 3000ms

【题目描述 - Problem Description】

X星球的某个大奖赛设了M级奖励。每个级别的奖金是一个正整数。

并且,相邻的两个级别间的比例是个固定值。

也就是说:所有级别的奖金数构成了一个等比数列。

比如: 16,24,36,54

其等比值为:3/2

现在,我们随机调查了一些获奖者的奖金数。

请你据此推算可能的最大的等比值。

输入格式:

第一行为数字N(N<=100),表示接下的一行包含N个正整数

第二行N个正整数Xi(Xi<1 000 000 000 000),用空格分开。每个整数表示调查到的某人的奖金数额

要求输出:

一个形如A/B的分数,要求A、B互质。表示可能的最大比例系数

测试数据保证了输入格式正确,并且最大比例是存在的。

输入 3
1250 200 32
4
3125 32 32 200
3
549755813888 524288 2
输出 25/4 5/2 4/1

【题解】

  等比数列求公比,Xi < 10^12

  先对各项排序,两两相除并约分,只要保证每次计算结果大于1,最后剩下的就是最大公比。

  N=100,O(N*N)还算可以接受。

  需要注意去重,以及公比为1的情况。

  数据范围10^12看起来诚惶诚恐,直接钦定long long可能会有人担心出现10^24的情况,(因为强迫症)于是就写了简单的证明……

设等比数列通项:

由于都是整数项,遇到分数形式的公比必然为整除。
因此z, m < 10^12

在第一轮除法中,首项被消除,只留下公比的任意次幂,小于1则分子分母互换。
运算中由于GCD的存在:

因此在第一轮的计算峰值不会超过10^12次方

在第2~n轮的计算中,由于去除了不确定的首项,可以把通项转化如下(其实一开始去掉首项也一样):

  两两相除时:

很遗憾,计算峰值还是无法超过10^12

【代码 C++】

 1 #include <cstdio>
 2 #include <algorithm>
 3 struct fs {
 4     __int64 fz, fm;
 5     bool operator == (const fs &B)const {
 6         if (B.fz != fz || B.fm != fm) return 0;
 7         return 1;
 8     }
 9 }data[105];
10 __int64 rd[105];
11 __int64 GCD(__int64 a, __int64 b) {
12     __int64 c;
13     while (c = a%b) a = b, b = c;
14     return b;
15 }
16 fs slv(fs a, fs b) {
17     if (a == b) return a;
18     __int64 gcd;
19     gcd = GCD(a.fz, b.fz);
20     a.fz /= gcd; b.fz /= gcd;
21     gcd = GCD(a.fm, b.fm);
22     a.fm /= gcd; b.fm /= gcd;
23     a.fz *= b.fm;
24     b.fz *= a.fm;
25     if (a.fz > b.fz) a.fm = b.fz;
26     else a.fm = a.fz, a.fz = b.fz;
27     return a;
28 }
29 int main() {
30     int i, j, n;
31     scanf("%d", &n);
32     for (i = 0; i < n; ++i) scanf("%I64d", rd + i), data[i].fm = 1;
33     std::sort(rd, rd + n);
34     for (i = j = 0; i < n; ++i) if (rd[i] != rd[i + 1]) data[j++].fz = rd[i];
35     for (n = j - 1; n; --n) {
36         for (i = 0; i < n; ++i) data[i] = slv(data[i], data[i + 1]);
37     }
38     if (j == 1) puts("1/1");
39     else printf("%I64d/%I64d", data[0].fz, data[0].fm);
40     return 0;
41 }
时间: 2024-08-28 01:27:24

[蓝桥杯] 最大比例的相关文章

2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告

2016 第七届蓝桥杯 c/c++ B组省赛真题及解题报告 勘误1:第6题第4个 if最后一个条件粗心写错了,答案应为1580. 条件应为abs(a[3]-a[7])!=1,宝宝心理苦啊.!感谢zzh童鞋的提醒. 勘误2:第7题在推断连通的时候条件写错了,后两个if条件中是应该是<=12 落了一个等于号.正确答案应为116. 1.煤球数目 有一堆煤球.堆成三角棱锥形.详细: 第一层放1个, 第二层3个(排列成三角形), 第三层6个(排列成三角形), 第四层10个(排列成三角形). -. 假设一共

2016年蓝桥杯C/C++B组

第一次参加蓝桥杯,也是有很多感触的,时间完全不够写最后一题... 最后一题没做...还有全排序很重要... 1. 煤球数目 有一堆煤球,堆成三角棱锥形.具体: 第一层放1个, 第二层3个(排列成三角形), 第三层6个(排列成三角形), 第四层10个(排列成三角形), -. 如果一共有100层,共有多少个煤球? 请填表示煤球总数目的数字. 注意:你提交的应该是一个整数,不要填写任何多余的内容或说明性文字. /*答案:171700*/ #include <iostream> #include &l

算法-蓝桥杯习题(二)

蓝桥杯习题 算法训练(111题) 1 /* 2 算法训练 明明的随机数 3 4 问题描述 5 明明想在学校中请一些同学一起做一项问卷调查,为了实验的客观性,他先用计算机生成了N个1到1000之间的随机整数(N≤100),对于其中重复的数字,只保留一个,把其余相同的数去掉,不同的数对应着不同的学生的学号.然后再把这些数从小到大排序,按照排好的顺序去找同学做调查.请你协助明明完成“去重”与“排序”的工作. 6 输入格式 7 输入有2行,第1行为1个正整数,表示所生成的随机数的个数: 8 N 9 第2

2015年蓝桥杯省赛B组C/C++(试题+答案)

首先说,这次我是第二次参加蓝桥杯(大学里最后一次),可这次去连个三等都没拿到,有些心灰意冷,比上一次还差, 当时看到成绩出来的时候有些失落,但是跌倒了,再站起来继续跑就可以了.可能是状态不好吧,纯属自我安慰. 接下来我把今年的题目又重新做了一遍,写下了这篇博客,如果也有需要探讨答案的,希望可以有帮助. 第一题: 第1题:统计不含4的数字 题目大意 统计10000至99999中,不包含4的数值个数. 解题分析: 第一种解法: 数学方法,这种是在网上看到的一种解法: 最高位除了0.4不能使用,其余8

蓝桥杯——算法训练之乘积最大

问题描述 今年是国际数学联盟确定的"2000--世界数学年",又恰逢我国著名数学家华罗庚先生诞辰90周年.在华罗庚先生的家乡江苏金坛,组织了一场别开生面的数学智力竞赛的活动,你的一个好朋友XZ也有幸得以参加.活动中,主持人给所有参加活动的选手出了这样一道题目: 设有一个长度为N的数字串,要求选手使用K个乘号将它分成K+1个部分,找出一种分法,使得这K+1个部分的乘积能够为最大. 同时,为了帮助选手能够正确理解题意,主持人还举了如下的一个例子: 有一个数字串:312, 当N=3,K=1时

蓝桥杯——判定字符的位置。

蓝桥杯——判断字符的位置 题目: 输入一个字符串,编写程序输出该字符串中元音字母的首次出现位置,如果没有元音字母输出0.英语元音字母只有‘a’.‘e’.‘i’.‘o’.‘u’五个. 样例输入: hello样例输出:2 样例输入: apple样例输出:1 样例输入: pmp样例输出:0 java code: import java.util.*;public class Yuanyingzifu {    public static int fun(String str)    {        

蓝桥杯 地宫取宝(12&#39;)

X 国王有一个地宫宝库.是n x m 个格子的矩阵.每个格子放一件宝贝.每个宝贝贴着价值标签. 地宫的入口在左上角,出口在右下角. 小明被带到地宫的入口,国王要求他只能向右或向下行走. 走过某个格子时,如果那个格子中的宝贝价值比小明手中任意宝贝价值都大, 小明就可以拿起它(当然,也可以不拿). 当小明走到出口时,如果他手中的宝贝恰好是k件,则这些宝贝就可以送给小明. 请你帮小明算一算,在给定的局面下,他有多少种不同的行动方案能获得这k件宝贝. [数据格式] 输入一行 3个整数,用空格分开:n m

蓝桥杯-开心的金明

//蓝桥杯-算法训练 开心的金明 //评测结果 AC //动态规划 01背包 #include <cstdio> #include <algorithm> #include <cstring> using namespace std; const int maxn = 30005; int main() { int N, m; scanf( "%d%d", &N, &m ); int f[maxn]; int v, w; memset

第六届蓝桥杯2015-省赛-C语言大学B组 个人题解

题目连接:http://course.baidu.com/view/2d86a6c1960590c69fc37622.html 1,奖券数目52488 #include <iostream> using namespace std; bool isOK(int a) { while(a) { if(a%10==4) return false; a /= 10; } return true; } int main() { int ans = 0; for(int i=10000; i<=9