洛谷 P1220 关路灯

题目描述
某一村庄在一条路线上安装了n盏路灯,每盏灯的功率有大有小(即同一段时间内消耗的电量有多有少)。老张就住在这条路中间某一路灯旁,他有一项工作就是每天早上天亮时一盏一盏地关掉这些路灯。
为了给村里节省电费,老张记录下了每盏路灯的位置和功率,他每次关灯时也都是尽快地去关,但是老张不知道怎样去关灯才能够最节省电。他每天都是在天亮时首先关掉自己所处位置的路灯,然后可以向左也可以向右去关灯。开始他以为先算一下左边路灯的总功率再算一下右边路灯的总功率,然后选择先关掉功率大的一边,再回过头来关掉另一边的路灯,而事实并非如此,因为在关的过程中适当地调头有可能会更省一些。
现在已知老张走的速度为1m/s,每个路灯的位置(是一个整数,即距路线起点的距离,单位:m)、功率(W),老张关灯所用的时间很短而可以忽略不计。
请你为老张编一程序来安排关灯的顺序,使从老张开始关灯时刻算起所有灯消耗电最少(灯关掉后便不再消耗电了)。
输入输出格式
输入格式:
文件第一行是两个数字n(0 < n < 50,表示路灯的总数)和c(1<=c<=n老张所处位置的路灯号);
接下来n行,每行两个数据,表示第1盏到第n盏路灯的位置和功率。
输出格式:
一个数据,即最少的功耗(单位:J,1J=1W·s)。
输入输出样例
输入样例#1:
5 3
2 10
3 20
5 20
6 30
8 10
输出样例#1:
270
说明
输出解释:
{此时关灯顺序为3 4 2 1 5,不必输出这个关灯顺序}

分析:

注意功率做一个前缀和
f[i][j][0]表示关掉i,j这段区间的灯,并且老张在区间的左边的最小花费
f[i][j][1]表示关掉i,j这段区间的灯,并且老张在区间的右边的最小花费
显然 f[i][j][0] 可以从 [i+1][j][0]向左走一个灯 也可以从[i+1][j][1]向左走过一段区间来到这里
转移方程:
f[i][j][0]=min((pos[i+1]-pos[i]) * (pst[n]-pst[j]+pst[i])+f[i+1][j][0],(pos[j]-pos[i]) * (pst[n]-pst[j]+pst[i])+f[i+1][j][1]);
f[i][j][1]=min((pos[j]-pos[j-1]) * (pst[n]-pst[j-1]+pst[i-1])+f[i][j-1][1],(pos[j]-pos[i]) * (pst[n]-pst[j-1]+pst[i-1])+f[i][j-1][0]);

#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
using namespace std;
#define MAXN 55
int f[MAXN][MAXN][2],n,c,pos[MAXN],pst[MAXN],Ans;

inline void read(int &x){
    x=0; int f=1; char c=getchar();
    while(c>'9'||c<'0'){ if(c=='-')f=-1; c=getchar(); }
    while(c>='0'&&c<='9'){ x=x*10+c-'0'; c=getchar(); } x*=f;
}

int Main(){
    read(n),read(c),pst[0]=0;
    for(int i=1;i<=n;++i){
        read(pos[i]);
        int x;read(x);
         pst[i]=pst[i-1]+x;
    }
    memset(f,0x3f,sizeof f );
    f[c][c][0]=f[c][c][1]=0;
    for(int k=1;k<n;++k){
        int s=max(0,c-k),t=min(n,k+c);
        for(int i=s;i<=t;++i){
            int j=i+k;
            f[i][j][0]=min((pos[i+1]-pos[i])*(pst[n]-pst[j]+pst[i])+f[i+1][j][0],(pos[j]-pos[i])*(pst[n]-pst[j]+pst[i])+f[i+1][j][1]);
            f[i][j][1]=min((pos[j]-pos[j-1])*(pst[n]-pst[j-1]+pst[i-1])+f[i][j-1][1],(pos[j]-pos[i])*(pst[n]-pst[j-1]+pst[i-1])+f[i][j-1][0]);
        }
    }
    Ans=min(f[1][n][0],f[1][n][1]);
    printf("%d\n",Ans);
    return 0;
}
int Aptal_is_My_Son=Main();
int main(int argc,char *argv[]){ ; }
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

七情六欲·

学生党不容易~

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

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

打赏作者

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

抵扣说明:

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

余额充值