算法复习之磁带最优储存问题

题目描述

设有n 个程序{1,2,…, n }要存放在长度为L的磁带上。程序i存放在磁带上的长度是Li, 1<= i<= n。这n 个程序的读取概率分别是p1,p2,…,pn,且pi+p2+…+pn = 1。如果将这n 个程序按 i1,i2,…,in 的次序存放,则读取程序ir 所需的时间tr=c*(Pi1Li2+Pi2Li2+…+Pir*Lir)。这n 个程序的平均读取 时间为t1+t2+…+tn。 磁带最优存储问题要求确定这n 个程序在磁带上的一个存储次序,使平均读取时间达到 最小。试设计一个解此问题的算法,并分析算法的正确性和计算复杂性。 编程任务: 对于给定的n个程序存放在磁带上的长度和读取概率,编程计算n个程序的最优存储方 案。

输入

由文件input.txt给出输入数据。第一行是正整数n,表示文件个数。接下来的n行中,
每行有2 个正整数a 和b, 分别表示程序存放在磁带上的长度和读取概率。 我觉得这里题目描述有错误,应该a是概率,b是长度
实际上第k个程 序的读取概率ak/(a1+a2+…+an)。对所有输入均假定c=1。

输出

输出一个实数,保留1位小数,表示计算出的最小平均读取时间。

示例输入

5
71 872
46 452
9 265
73 120
35 87

示例输出

85.619

算法设计

求解最短读取时间,可用贪心算法求解该问题,

  1. 先计算每个程序使用的概率
  2. 通过公式,计算每个程序的pi*li
  3. 将其递增排序(贪心求解)
  4. 用公式求解最优解
    (应该证明他具有贪心选择性质,可惜我不会,我赌他不考证明…)

代码


#include<stdio.h>
#include<iostream>
using namespace std;

int Print(double a[],int n)
{
    for(int i=0;i<n;i++)
    {
        cout<<a[i]<<" ";
    }
    cout<<endl;
}


void Proba(double m[],int a[],int n)
{
    int sum=0;
    for(int i=0;i<n;i++)
    {
        sum+=a[i];
    }
    for(int i=0;i<n;i++)
    {
        m[i]=(double)a[i]/sum;
    }

}
void calTime(double m[],double p[],int length[],int n)
{
   // m[0]=p[0]*length[0];
    for(int i=0;i<n ;i++)
    {
        m[i]=p[i]*length[i];
    }
    Print(m,n);


}
void sort(double m[],int n)
{
    double t;
    for(int i=0;i<n;i++)
    {
        for(int j=i;j<n;j++)
        {
            if(m[j]<m[i])
            {
                t=m[i];
                m[i]=m[j];
                m[j]=t;
            }
        }
    }
    Print(m,n);
}
double BestTime(double Evetime[],int n)
{
    double TR[n];
    TR[0]=Evetime[0];
    for(int i=1;i<n;i++)
    {
        TR[i]=TR[i-1]+Evetime[i];
    }

    double sum=0;
    for(int i=0;i<n;i++)
    {
        sum+=TR[i];
    }
    return sum;
}

double MinTime(int a[],int b[],int n)
{
    double prob[n];
    Proba(prob,a,n);
    Print(prob,n);
    double Evetime[n];
    calTime(Evetime,prob,b,n);
    sort(Evetime,n);
    return BestTime(Evetime,n);
}

int main()
{
    //为了简单,没从文件读取,直接键盘输入了
    int n;
    scanf("%d",&n);
    int a[n];
    int b[n];
    for(int i=0;i<n;i++)
    {
        scanf("%d %d",a+i,b+i);
    }
    double t;
    t=MinTime(b,a,n);
    printf("最优时间为:%.3lf",t);
}

输出结果

在这里插入图片描述

  • 3
    点赞
  • 20
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值