[CODEVS1258]关路灯

 题目描述 Description

多瑞卡得到了一份有趣而高薪的工作。每天早晨他必须关掉他所在村庄的街灯。所有的街灯都被设置在一条直路的同一侧。

多瑞卡每晚到早晨5点钟都在晚会上,然后他开始关灯。开始时,他站在某一盏路灯的旁边。

每盏灯都有一个给定功率的电灯泡,因为多端卡有着自觉的节能意识,他希望在耗能总数最少的情况下将所有的灯关掉。

多端卡因为太累了,所以只能以1m/s的速度行走。关灯不需要花费额外的时间,因为当他通过时就能将灯关掉。

编写程序,计算在给定路灯设置,灯泡功率以及多端卡的起始位置的情况下关掉所有的灯需耗费的最小能量。

输入描述 Input Description

输入文件的第一行包含一个整数N,2≤N≤1000,表示该村庄路灯的数量。

第二行包含一个整数V,1≤V≤N,表示多瑞卡开始关灯的路灯号码。

接下来的N行中,每行包含两个用空格隔开的整数D和W,用来描述每盏灯的参数,其中0≤D≤1000,0≤W≤1000。D表示该路灯与村庄开始处的距离(用米为单位来表示),W表示灯泡的功率,即在每秒种该灯泡所消耗的能量数。路灯是按顺序给定的。

输出描述 Output Description

输出文件的第一行即唯一的一行应包含一个整数,即消耗能量之和的最小值。注意结果小超过1,000,000,000。

样例输入 Sample Input

4

3

2 2

5 8

6 1

8 7

样例输出 Sample Output

56

思路

对于小数据来说,爆搜可以过,n<=50,然而对于大的数据,当n<=1000时,只能过7个点。

设f[i,j,k](k=1或2)表示已经关过的MM区间为[i,j],k=1表示当前在左端点i位置,k=2表示当前在右端点j位置。

   f[i,j,0]=min{ f[i+1,j,1] + (w[i] + (w[n] - w[j]))*(d[i+1] - d[i]),
            f[i+1,j,2] + (w[i]+ (w[n] - w[j]))*(d[i+1] - d[i]) }
  f[i,j,2]=min{ f[i,j-1,1] + (w[i-1] + (w[n] - w[j-1]))*(d[j] - d[i]),
          f[i,j-1,2] + (w[i-1] + (w[n] - w[j-1]))*(d[j] - d[j-1]) }
 初始状态:f[s,s,1]=s[s,s,2]=0
 目标状态:min{ f[1,n,1] , f[1,n,2] }
 时间复杂度:O(n^2)
DFS
var
  n,c,ans,total:longint;
  dis,w:array[0..60]of longint;
  f:array[0..60]of boolean;

procedure init;
begin
  assign(input,'power.in');
  assign(output,'power.out');
  reset(input);
  rewrite(output);
end;

procedure terminate;
begin
  close(input); close(output);
  halt;
end;

procedure dfs(t,tot,total,t1:longint);
var
  i:longint;
begin
  for i:=t-1 downto 1 do
    if f[i] then
      begin
        if tot+(dis[t]-dis[i])*(total-w[t])<ans then
          begin
            f[i]:=false;
            dfs(i,tot+(dis[t]-dis[i])*(total-w[t]),total-w[t],t1+1);
            f[i]:=true;
            break;
          end;
      end;
  for i:=t+1 to n do
    if f[i] then
      begin
        if tot+(dis[i]-dis[t])*(total-w[t])<ans then
          begin
            f[i]:=false;
            dfs(i,tot+(dis[i]-dis[t])*(total-w[t]),total-w[t],t1+1);
            f[i]:=true;
            break;
          end;
      end;
  if t1=n then
    if tot<ans then
      begin
        ans:=tot;
        exit;
      end;
end;

procedure main;
var
  i:longint;
begin
  readln(n,c);
  total:=0;
  for i:=1 to n do
    begin
      readln(dis[i],w[i]);
      f[i]:=true;
      total:=total+w[i];
    end;
  ans:=maxlongint;
  f[c]:=false;
  dfs(c,0,total,1);
  writeln(ans);
end;

begin
  init;
  main;
  terminate;
end.
View Code

 

DP

program CloseTheLight;
uses math;
var n,c,i,p,j,k,v1,a:longint;
    v,s:array[0..1000] of longint;
    f:array[0..1000,0..1000,0..1] of longint;
begin
    read(n,c);
    for i:=1 to n do
    begin
        read(v[i],a);
        s[i]:=s[i-1]+a;
    end;
    filldword(f,sizeof(f) div 4,maxlongint);
    f[c,c,1]:=0;
    f[c,c,0]:=0;
    for p:=0 to n-1 do
        for i:=1 to n-p do
        begin
            j:=i+p;
            for k:=0 to 1 do
            begin
                if k=0 then v1:=i else v1:=j;
                if f[i,j,k]<maxlongint then
                begin
                    if i>0 then f[i-1,j,0]:=min(f[i-1,j,0],
                                            f[i,j,k]+(s[i-1]+s[n]-s[j])*(v[v1]-v[i-1]));
                    if j<n then f[i,j+1,1]:=min(f[i,j+1,1],
                                            f[i,j,k]+(s[i-1]+s[n]-s[j])*(v[j+1]-v[v1]));
                end;
            end;
        end;
    writeln(min(f[1,n,0],f[1,n,1]));
end.
View Code

 

posted on 2015-09-08 20:41  川汉唐  阅读(247)  评论(0编辑  收藏  举报

导航