zoukankan      html  css  js  c++  java
  • bzoj4038: 医疗援助

    Description

    一只带着先进设备和药物的医疗团队来到了埃博拉病毒疫区的某个非洲国家。这个国家有n个村庄,均坐落在该国唯一的一条公路旁,n个村庄依次标号为1,2,…n。第i个村庄有a_i个埃博拉感染者。
    到来后的第一天早晨,医疗团队在第1个村庄。每天他们可以选择治疗所在村庄的村民,这样所有感染者都会痊愈;他们也可以选择前往相邻的一个村庄,路程需要1天。每天结束时,如果第i个村庄的a_i个感染者没有痊愈,那么他们会死去,同时会有另外a_i个人被感染。也就是说,如果第i个村庄没有被治疗,那么每天这个村庄会死去a_i个人。
    医疗团队在经过村庄i时,可能选择不治疗这个村庄而前往下一个村庄i+1。但为了不让村民失去希望,如果医疗团队在村庄j决定往回走时,则他们要治疗村庄j之前所有没有被治疗的村庄,同时在返回的过程中,如果经过没有接受治疗的村庄,他们需要停下来进行治疗。
    医疗团队最终会治愈所有村民。作为团队的领导人,你需要得出在合理规划行程下最少的死亡人数。
     
     

    Input

    第一行为一个正整数n,表示村庄的个数。接下来一行是n个正整数a_i(1<=a_i<=10^9),为每个村庄的感染人数

    Output

    输出一个整数,表示在治愈所有村民前最少的死亡人数。

    Sample Input

    6
    40 200 1 300 2 10

    Sample Output

    1950

    HINT

    100%的数据,n<=3000

    题解:
    设g[i][j]表示从j开始走,走到i,然后回头到j,i到j之间的村子的最小死亡人数
    显然g[i][i]=0
    考虑j这个村子一开始是治疗还是被跳过
    所以g[i][j]=g[i][j+1]+sum[j+1,i]+min((i-j)*3*a[j],sum[j+1,i])
    然后设f[i]表示走到i,且前面的都治好了的情况下,总共的最小死亡人数
    转移就是枚举j,考虑从j+1出发到i,然后回头到j+1,再回头到i
    即f[i]=min{f[j]+g[i][j+1]+sum[i+1,n]*((i-j)*4-2)}
    复杂度O(n2)
    PS:这也是jsoi2016R2D2T1
    code:
     1 #include<cstdio>
     2 #include<iostream>
     3 #include<cmath>
     4 #include<cstring>
     5 #include<algorithm>
     6 using namespace std;
     7 typedef long long int64;
     8 char ch;
     9 bool ok;
    10 void read(int &x){
    11     for (ok=0,ch=getchar();!isdigit(ch);ch=getchar()) if (ch=='-') ok=1;
    12     for (x=0;isdigit(ch);x=x*10+ch-'0',ch=getchar());
    13     if (ok) x=-x;
    14 }
    15 void read(int64 &x){
    16     for (ok=0,ch=getchar();!isdigit(ch);ch=getchar()) if (ch=='-') ok=1;
    17     for (x=0;isdigit(ch);x=x*10+ch-'0',ch=getchar());
    18     if (ok) x=-x;
    19 }
    20 const int maxn=3005;
    21 int n;
    22 int64 a[maxn],sum[maxn],g[maxn][maxn],f[maxn];
    23 inline int64 s(int l,int r){return sum[r]-sum[l-1];}
    24 int main(){
    25     read(n);
    26     for (int i=1;i<=n;i++) read(a[i]);
    27     for (int i=1;i<=n;i++) sum[i]=sum[i-1]+a[i];
    28     for (int i=1;i<=n;i++){
    29         g[i][i]=0;
    30         for (int j=i-1;j>=1;j--) g[i][j]=g[i][j+1]+s(j+1,i)+min(3LL*(i-j)*a[j],s(j+1,i));
    31     }
    32     memset(f,63,sizeof(f));
    33     f[0]=0;
    34     for (int i=1;i<=n;i++) for (int j=0;j<i;j++) f[i]=min(f[i],f[j]+g[i][j+1]+s(i+1,n)*(4LL*(i-j)-2));
    35     printf("%lld
    ",f[n]);
    36     return 0;
    37 }
  • 相关阅读:
    最近的3个困惑:信守承诺、技术产品先行还是市场销售先行、客户从哪来
    最近的3个困惑:信守承诺、技术产品先行还是市场销售先行、客户从哪来
    详细回复某个CSDN网友,对我的文章和技术实力以及CSDN的吐槽
    详细回复某个CSDN网友,对我的文章和技术实力以及CSDN的吐槽
    2015年工作中遇到的问题:21-30(这10个问题很有价值)
    使用ABAP(ADBC)和Java(JDBC)连接SAP HANA数据库
    C4C和Outlook的集成
    Hybris开发环境的license计算实现
    CRM WebClient UI和Hybris里工作中心跳转的url生成逻辑
    CRM WebUI and Hybris的Product页面标题实现
  • 原文地址:https://www.cnblogs.com/chenyushuo/p/5583582.html
Copyright © 2011-2022 走看看