2015湘潭邀请赛 D.Fraction

14 篇文章 0 订阅

刻骨铭心的一道题
http://202.197.224.59/OnlineJudge2/index.php/Problem/read/id/1236

当时比赛的时候就是卡在了这道题目上,大sb题,其实当时就是把题目的条件理解多了一层意思
If several fractions satifies the restriction, he chooses the smallest one with simplest formation.
就是这句话,当时理解的是取的是值最小且误差最小,但实际这种语法指的就是误差最小!!!坑爹啊,现场赛时候最后都没意识到这点,还浪费了那么多的时间!

#include<bits/stdc++.h>
using namespace std;
double x;
int main(){
    int t;cin>>t;int a,b;
    while(t--){
        scanf("%lf",&x);double minn=100000000.0;
        for(int i=1;i<=1000;i++){
            double ans=i*x;
            int d=(int)ans;
            if(fabs((double)d/(double)i-x)<minn){
                minn=fabs((double)d/(double)i-x);
                a=d;b=i;
            }
            if(fabs((double)(d+1)/(double)i-x)<minn){
                minn=fabs((double)(d+1)/(double)i-x);
                a=d+1;b=i;
            }
        }
        printf("%d/%d\n",a,b);
    }
}

也可以这么做:
思路:分母不大于1000,直接预处理出所有的a/b,gcd(a,b) = 1,排序。对于输入的每个分数,直接二分查找,最接近的肯定是不小于它的最小的那个分数或者不大于它的最大的那个分数

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <cstdlib>
using namespace std;

const int N = 1010;
struct point
{
    int x,y;
    double z;
    friend bool operator < (const point &p,const point &q) {
        return p.z < q.z;
    }
}a[N * N];

int main()
{
    int cnt = 0;
    for(int i = 1; i <= 1000; i ++) {
        for(int j = 0; j <= i; j ++) {
            if(__gcd(i,j) != 1) continue;
            a[cnt].x = j,a[cnt].y = i,a[cnt].z = 1.0 * j / i;
            cnt ++;
        }
    }
    sort(a,a + cnt);
    int t;
    scanf("%d",&t);
    while(t --) {
        double x;
        scanf("%lf",&x);
        int idx = cnt - 1,lt = 0,rt = cnt - 1,mid;
        while(lt <= rt) {
            mid = lt + rt >> 1;
            if(a[mid].z >= x) {
                idx = mid;
                rt = mid - 1;
            }
            else lt = mid + 1;
        }
        if(fabs(a[idx - 1].z - x) < fabs(a[idx].z - x)) {
            printf("%d/%d\n",a[idx - 1].x,a[idx - 1].y);
        }
        else printf("%d/%d\n",a[idx].x,a[idx].y);
    }
    return 0;
}
  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值