NOIP2016换教室

题目描述 Description

对于刚上大学的牛牛来说,他面临的第一个问题是如何根据实际情况申请合适的课程。在可以选择的课程中,有\(2n\) 节课程安排在\(n\) 个时间段上。在第\(i(1≤i≤n)\) 个时间段上,两节内容相同的课程同时在不同的地点进行,其中,牛牛预先被安排在教室\(c_i\) 上课,而另一节课程在教室\(d_i\) 进行。在不提交任何申请的情况下,学生们需要按时间段的顺序依次完成所有的\(n\) 节安排好的课程。如果学生想更换第\(i\) 节课程的教室,则需要提出申请。若申请通过,学生就可以在第\(i\) 个时间段去教室\(d_i\) 上课,否则仍然在教室\(c_i\)上课。由于更换教室的需求太多,申请不一定能获得通过。通过计算,牛牛发现申请更换第i节课程的教室时,申请被通过的概率是一个已知的实数\(k_i\) ,并且对于不同课程的申请,被通过的概率是互相独立的。学校规定,所有的申请只能在学期开始前一次性提交,并且每个人只能选择至多\(m\) 节课程进行申请。这意味着牛牛必须一次性决定是否申请更换每节课的教室,而不能根据某些课程的申请结果来决定其他课程是否申请;牛牛可以申请自己最希望更换教室的\(m\) 门课程,也可以不用完这\(m\) 个申请的机会,甚至可以一门课程都不申请。因为不同的课程可能会被安排在不同的教室进行,所以牛牛需要利用课间时间从一间教室赶到另一间教室。牛牛所在的大学有v个教室,有\(e\) 条道路。每条道路连接两间教室,并且是可以双向通行的。由于道路的长度和拥堵程度不同,通过不同的道路耗费的体力可能会有所不同。当第\(i(1≤i≤n-1)\) 节课结束后,牛牛就会从这节课的教室出发,选择一条耗费体力最少的路径前往下一节课的教室。现在牛牛想知道,申请哪几门课程可以使他因在教室间移动耗费的体力值的总和的期望值最小,请你帮他求出这个最小值。

输入描述 Input Description

第一行四个整数\(n,m,v,e\) 。n表示这个学期内的时间段的数量;\(m\) 表示牛牛最多可以申请更换多少节课程的教室;\(v\) 表示牛牛学校里教室的数量;\(e\) 表示牛牛的学校里道路的数量。
第二行n个正整数,第\(i(1≤i≤n)\) 个正整数表示\(c\) ,,即第\(i\) 个时间段牛牛被安排上课的教室;保证\(1≤c_i≤v\)
第三行n个正整数,第\(i(1≤i≤n)\) 个正整数表示\(d_i\),即第\(i\) 个时间段另一间上同样课程的教室;保证\(1≤d_i≤v\)
第四行n个实数,第\(i(1≤i≤n)\) 个实数表示\(k_i\),即牛牛申请在第\(i\) 个时间段更换教室获得通过的概率。保证\(0≤k_i≤1\)
接下来e行,每行三个正整数\(a_j,b_j,w_j\),表示有一条双向道路连接教室\(a_j,b_j\) ,通过这条道路需要耗费的体力值是\(W_j\);

输出描述 Output Description

输出一行,包含一个实数,四舎五入精确到小数点后恰好2位,表示答案。你的输出必须和标准输出完全一样才算正确。
测试数据保证四舎五入后的答案和准确答案的差的绝对值不大于\(4*10^{-3}\) 。(如果你不知道什么是浮点误差,这段话可以理解为:对于大多数的算法,你可以正常地使用浮点数类型而不用对它进行特殊的处理)

样例输入 Sample Input

3 2 3 3
2 1 2
1 2 1
0.8 0.2 0.5 
1 2 5
1 3 3
2 3 1

样例输出 Sample Output

2.80

数据范围及提示 Data Size & Hint

保证\(1≤a_j,b_j≤v,1≤w_j≤100。\)
保证\(1≤n≤2000,0≤m≤2000,1≤v≤300,0≤e≤90000。\)
保证通过学校里的道路,从任何一间教室出发,都能到达其他所有的教室。
保证输入的实数最多包含\(3\) 位小数。

之前的一些废话

题解

首先\(floyd\) 一遍,方便我们转移,然后:
\(dp[i][j][0]表示到了第i节课,目前已经申请了j次,0或1表示当前这次换不换\)
$ 转移为 dp[i+1][j][0]=min ( dp[i][j][0]+e[c_i][c_{i+1}],dp[i][j][1]+k_i * e[d_i][c_{i+1}]+(1-k_{i}) * e[c_i][c_{i+1}] ) $
$ dp[i+1][j+1][1]=min ( dp[i][j][0]+k_{i+1} * e[c_i][d_{i+1}]+(1-k_{i+1}) * e[c_i][c_{i+1}], $
$ dp[i][j][1]+k_i * k_{i+1} * e[d_i][d_{i+1}]+k_i * (i-k_{i+1}) * e[d_i][c_{i+1}] +(1-k_i) * k_{i+1} * e[c_i][d_{i+1}]+(1-k_i) * (1-k_{i+1}) * e[c_i][c_{i+1}] ) $ ,之后转移即可。

代码

#include
#include
#include
#include
#include
#include
using namespace std;
typedef long long LL;
typedef pair PII;
#define mem(a,b) memset(a,b,sizeof(a))
inline int read()
{
    int x=0,f=1;char c=getchar();
    while(!isdigit(c)){if(c=='-')f=-1;c=getchar();}
    while(isdigit(c)){x=x*10+c-'0';c=getchar();}
    return x*f;
}
const int maxn=2010,maxN=310;
int n,m,N,M,c[maxn],d[maxn],e[maxN][maxN],x,y,z;
double k1[maxn],k2[maxn],dp[maxn][maxn][2],ans=1e20;
int main()
{
//  freopen("flame.in","r",stdin);
//  freopen("flame.out","w",stdout);
    mem(e,42);mem(dp,42); 
    n=read();m=read();N=read();M=read();
    for(int i=1;i<=n;i++)c[i]=read();
    for(int i=1;i<=n;i++)d[i]=read();
    for(int i=1;i<=n;i++)scanf("%lf",&k1[i]),k2[i]=1.0-k1[i];//,cout<<"fa"; 
    for(int i=1;i<=N;i++)e[i][i]=0;
    for(int i=1;i<=M;i++)
    {
        x=read();y=read();z=read();
        e[x][y]=min(e[x][y],z);
        e[y][x]=min(e[y][x],z);
    }
    for(int k=1;k<=N;k++)
        for(int i=1;i<=N;i++)
            for(int j=1;j<=N;j++)e[i][j]=min(e[i][j],e[i][k]+e[k][j]);
    for(int i=1;i<=n;i++)for(int j=0;j<=m;j++)dp[i][j][0]=dp[i][j][1]=1e20;
    dp[1][0][0]=0.0;dp[1][1][1]=0.0;
    for(int i=1;i<=n;i++)
        for(int j=0;j<=min(i,m);j++)
        {
            dp[i+1][j][0]=min(dp[i+1][j][0],dp[i][j][0]+e[c[i]][c[i+1]]);
            if(j>0)dp[i+1][j][0]=min(dp[i+1][j][0],dp[i][j][1]+k1[i]*e[d[i]][c[i+1]]+k2[i]*e[c[i]][c[i+1]]);
            dp[i+1][j+1][1]=min(dp[i+1][j+1][1],dp[i][j][0]+k1[i+1]*e[c[i]][d[i+1]]+k2[i+1]*e[c[i]][c[i+1]]);
            if(j>0)dp[i+1][j+1][1]=min(dp[i+1][j+1][1],dp[i][j][1]+k1[i]*k1[i+1]*e[d[i]][d[i+1]]+
                                                                   k1[i]*k2[i+1]*e[d[i]][c[i+1]]+
                                                                   k2[i]*k1[i+1]*e[c[i]][d[i+1]]+
                                                                   k2[i]*k2[i+1]*e[c[i]][c[i+1]]); 
        }
    ans=dp[n][0][0];
    for(int i=1;i<=m;i++)ans=min(ans,min(dp[n][i][0],dp[n][i][1]));
    printf("%.2lf\n",ans);
    return 0;
}

总结

DP一定要从合法的地方转移过来,注意这么恶心的转移方程敲到键盘上时千万不要手残哦!

转载于:https://www.cnblogs.com/FYH-SSGSS/p/7813674.html

你可能感兴趣的:(数据结构与算法)