题目

一道毒瘤概率期望DP。点这里

首先

对于每个时间段(就是每节课),我们有申请和不申请两种情况。如果不申请的话,一定在\(c[ i ]\),否则,可能在\(c[ i ]\),也可能在\(d[ i ]\)。所以我们先预处理一些东西。 先用弗洛伊德处理出最短路(这就不用详细介绍了吧)。我们先考虑相邻的两间教室,如果这两间教室都申请的话,就会有四种情况(①都通过,②都没通过,③第一个通过,④第二个通过)感觉又长又臭。所以我们先处理出两间教室都申请的情况。 用\(a[ i ]\)表示在都申请的前提下从第i个教室走到第i+1个教室的期望长度 \(dis[ i ] [ j ]\)表示从教室i到j的最短路。

情况①的概率为 $ k[ i ] k[ i+1 ] $ ,再乘上路径的长度 \(dis[ c[ i ] ][ c[ i+1 ] ]\)

同理 情况② \(1 - k[ i ]) (1 - k[ i+1 ]) dis[ d[ i ] ][ d[ i+1 ] ]\)

情况③ \(k[ i ] (1 - k[ i+1 ]) dis[ c[ i ] ][ d[ i+1 ] ]\)

情况④$(1 - k[ i ]) k[ i+1 ] * dis[ d[ i ] ][ c[ i+1 ] ] $

所以:

for(int i=1;i<=n;i++)
a[i]=(1-k[i])*(1-k[i+1])*dis[c[i]][c[i+1]]+
k[i]*(1-k[i+1])*dis[d[i]][c[i+1]]+k[i]*k[i+1]*dis[d[i]][d[i+1]]+
(1-k[i])*k[i+1]*dis[c[i]][d[i+1]]

太丑了

然后,继续预处理......

\(dp[ i ][ 0/1 ][ 0/1 ]\) 表示从第i节课的教室到第i+1节课的教室,第i节课的教室有(1)/没有(0)申请,第i+1个教室有(1)/没有(0)申请的期望长度

然后\(dp[ i ][ 1 ][ 1 ]\)的情况就是刚才的\(a[ i ]\)数组啦

其余情况同理可得,于是就:

for(int i=1;i<n;i++)
{
dp[i][0][0]=dis[c[i]][c[i+1]];
dp[i][1][1]=a[i];
dp[i][0][1]=k[i+1]*dis[c[i]][d[i+1]]+(1-k[i+1])*dis[c[i]][c[i+1]];
dp[i][1][0]=k[i]*dis[d[i]][c[i+1]]+(1-k[i])*dis[c[i]][c[i+1]];
}

然后,终于开始DP啦

用\(f[ i ][ j ][ 0/1 ]\) 示到达前i个教室,申请了j次,第i间教室有没有申请的期望最短长度。

!!!注意,j要从0开始,因为可以一次也不申请!!!

\(f[ i ][ j ][ 0 ]\)可以由\(f[ i - 1 ][ j ][ 0 ]\)和\(f[ i - 1 ][ j ][ 1 ]\)转移过来

\(f[ i ][ j ][ 1 ]\)可以由\(f[ i - 1][ j - 1 ][ 0 ]\)和\(f[ i - 1][ j - 1 ][ 1 ]\)转移过来

先初始化一下 \(f[ i ][ 0 ][ 0 ]\) 的情况

于是就得到了:

   f[1][1][1]=0; f[1][0][0]=0;
for(int i=2;i<=n;i++)
f[i][0][0]=f[i-1][0][0]+dis[c[i-1]][c[i]];
for(int i=2;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
f[i][j][0]=min(f[i-1][j][1]+dp[i-1][1][0],f[i-1][j][0]+dp[i-1][0][0]);
f[i][j][1]=min(f[i-1][j-1][1]+dp[i-1][1][1],f[i-1][j-1][0]+dp[i-1][0][1]);
}
}

最后统计答案就行啦,要注意可以申请 0到m次。


for(int i=1;i<=m;i++)
ans=min(ans,min(f[n][i][0],f[n][i][1]));

于是,这道题就做完了!

最后贴上完整版代码(大佬勿喷):


#include<iostream>
#include<cstdio>
#include<cstring>
using namespace std;
const int N = 2005;
const int INF = 27654134;
int n,m,v,e,c[N],d[N],dis[N][N];
double a[N],k[N],f[N][N][2],dp[N][2][2],ans=99999999.9;
//a[i]表示在都申请的前提下从第i个教室走到第i+1个教室的期望长度
// dp[i][0/1][0/1]从第i个教室到第i+1个教室,第i个教室有没有申请,第i+1个教室有没有申请的期望长度
//f[i][j][0/1]表示到达前i个教室,申请了j次,第i间教室有没有申请的期望最短长度
int main()
{
memset(f,0x7f,sizeof(f));
// memset(dis,0x7f,sizeof(dis));
scanf("%d%d%d%d",&n,&m,&v,&e);
for(int i=1;i<=v;i++)
for(int j=1;j<=v;j++)
dis[i][j]=INF;
for(int i=1;i<=n;i++) scanf("%d",&c[i]);
for(int i=1;i<=n;i++) scanf("%d",&d[i]);
for(int i=1;i<=n;i++) cin>>k[i];
int x,y,z;
for(int i=1;i<=e;i++)
{
scanf("%d%d%d",&x,&y,&z);
if(dis[x][y]) dis[x][y]=min(dis[x][y],z),dis[y][x]=min(dis[y][x],z);
else dis[x][y]=z,dis[y][x]=z;
}
for(int i=1;i<=v;i++) dis[i][i]=0;
for(int kk=1;kk<=v;kk++)
{
for(int i=1;i<=v;i++)
{
if(kk==i) continue;
for(int j=1;j<=v;j++)
{
if(kk==j||i==j) continue;
if(dis[i][kk]+dis[kk][j]<dis[i][j])
dis[i][j]=dis[i][kk]+dis[kk][j];
}
}
}
for(int i=1;i<=n;i++)
a[i]=(1-k[i])*(1-k[i+1])*dis[c[i]][c[i+1]]+
k[i]*(1-k[i+1])*dis[d[i]][c[i+1]]+k[i]*k[i+1]*dis[d[i]][d[i+1]]+
(1-k[i])*k[i+1]*dis[c[i]][d[i+1]];
for(int i=1;i<n;i++)
{
dp[i][0][0]=dis[c[i]][c[i+1]];
dp[i][1][1]=a[i];
dp[i][0][1]=k[i+1]*dis[c[i]][d[i+1]]+(1-k[i+1])*dis[c[i]][c[i+1]];
dp[i][1][0]=k[i]*dis[d[i]][c[i+1]]+(1-k[i])*dis[c[i]][c[i+1]];
}
f[1][1][1]=0; f[1][0][0]=0;
for(int i=2;i<=n;i++)
f[i][0][0]=f[i-1][0][0]+dis[c[i-1]][c[i]];
for(int i=2;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
f[i][j][0]=min(f[i-1][j][1]+dp[i-1][1][0],f[i-1][j][0]+dp[i-1][0][0]);
f[i][j][1]=min(f[i-1][j-1][1]+dp[i-1][1][1],f[i-1][j-1][0]+dp[i-1][0][1]);
}
}
for(int i=0;i<=m;i++)
ans=min(ans,min(f[n][i][0],f[n][i][1]));
printf("%.2lf\n",ans);
return 0;
}

NOIP2016换教室的更多相关文章

  1. [NOIP2016]换教室 D1 T3 Floyed+期望DP

    [NOIP2016]换教室 D1 T3 Description 对于刚上大学的牛牛来说, 他面临的第一个问题是如何根据实际情况中情合适的课程. 在可以选择的课程中,有2n节课程安排在n个时间段上.在第 ...

  2. BZOJ 4720 [Noip2016]换教室

    4720: [Noip2016]换教室 Description 对于刚上大学的牛牛来说,他面临的第一个问题是如何根据实际情况申请合适的课程.在可以选择的课程中,有2n节课程安排在n个时间段上.在第i( ...

  3. 【BZOJ】4720: [Noip2016]换教室

    4720: [Noip2016]换教室 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1690  Solved: 979[Submit][Status ...

  4. bzoj4720: [Noip2016]换教室(期望dp)

    4720: [Noip2016]换教室 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 1294  Solved: 698[Submit][Status ...

  5. [NOIP2016]换教室 题解(奇怪的三种状态)

    2558. [NOIP2016]换教室 [题目描述] 对于刚上大学的牛牛来说,他面临的第一个问题是如何根据实际情况申请合适的课程. 在可以选择的课程中,有2n节课程安排在n个时间段上.在第i(1< ...

  6. 【bzoj4720】[NOIP2016]换教室

    题目描述 对于刚上大学的牛牛来说,他面临的第一个问题是如何根据实际情况申请合适的课程.在可以选择的课程中,有2n节课程安排在n个时间段上.在第i(1≤i≤n)个时间段上,两节内容相同的课程同时在不同的 ...

  7. [NOIp2016] 换教室

    题目类型:期望\(DP\) 传送门:>Here< 题意:现有\(N\)个时间段,每个时间段上一节课.如果不申请换教室,那么时间段\(i\)必须去教室\(c[i]\)上课,如果申请换课成功, ...

  8. [NOIP2016]换教室(概率期望$DP$)

    其实吧我老早就把这题切了--因为说实话,这道题确实不难啊--李云龙:比他娘的状压DP简单多了 今天我翻以前在Luogu上写的题解时,突然发现放错代码了,然后被一堆人\(hack\)--蓝瘦啊\(ORZ ...

  9. 【bzoj4720】[NOIP2016]换教室 期望dp

    题目描述 对于刚上大学的牛牛来说,他面临的第一个问题是如何根据实际情况申请合适的课程.在可以选择的课程中,有2n节课程安排在n个时间段上.在第i(1≤i≤n)个时间段上,两节内容相同的课程同时在不同的 ...

  10. NOIP2016换教室 BZOJ 4720

    BZOJ 4720 换教室 题目描述: 对于刚上大学的牛牛来说,他面临的第一个问题是如何根据实际情况申请合适的课程.在可以选择的课程中,有2n节 课程安排在n个时间段上.在第i(1≤i≤n)个时间段上 ...

随机推荐

  1. 远程连接腾讯云服务器MySQL数据库

    1.添加腾讯云安全组规则的MySQL 3306端口 将所有端口打开,至少打开3306,不在赘述. 2.打开更改MySQL配置文件 打开配置文件 vi /etc/mysql/mysql.conf.d/m ...

  2. 控制结构(8): 线性化(linearization)

    // 上一篇:管道(pipeline) // 下一篇:程序计数器(PC) "编程语言不过是一个工具,什么语言都一样","编程语言能改变人的思维,不同的语言会带给你不同的思 ...

  3. Kubernetes — Job与CronJob

    有一类作业显然不满足这样的条件,这就是“离线业务”,或者叫作 Batch Job(计算业务). 这 种业务在计算完成后就直接退出了,而此时如果你依然用 Deployment 来管理这种业务的话,就会 ...

  4. 《Effective C++》模板与泛型编程:条款32-条款40

    条款41:了解隐式接口和编译期多态 class支持显示接口和运行期多态 class的显示接口由函数的名签式构成(函数名称.参数类型.返回类型) class的多态通过virtual函数发生在运行期 te ...

  5. BEX5下增加sessionStorage监听器实现页面间数据刷新

    场景: A页面修改了数据,希望B页面能进行及时的同步前端数据,但是假如当A页面修改保存后,去获得B页面的model对象,会增加开发的难度,同时A页面也不能重复利用:假如在B页面的激活事件里面写刷新代码 ...

  6. Monkey参数介绍

    monkey 参数 参数分类 常规类参数 事件类参数 约束类参数 调试类参数 常规类参数 常规类参数包括帮助参数和日志信息参数.帮助参数用于输出Monkey命令使用指导:日志信息参数将日志分为三个级别 ...

  7. LODOP直线px转换mm变斜线

    LODOP中打印项顶边距左边距,宽高,可以选择的单位很多,详细可在LODOP官网下载参考LODOP技术手册. 关于LODOP打印直线和虚线,可查看本博客相关博文:Lodop如何打印直线.Lodop打印 ...

  8. CentOS 7安装MongoDB

    1 下载安装包 wget https://fastdl.mongodb.org/linux/mongodb-linux-x86_64-rhel70-3.2.4.tgz 2 解压 .tgz 3 将解压包 ...

  9. Dijkstra算法——计算一个点到其他所有点的最短路径的算法

    迪杰斯特拉算法百度百科定义:传送门 gh大佬博客:传送门 迪杰斯特拉算法用来计算一个点到其他所有点的最短路径,是一种时间复杂度相对比较优秀的算法 O(n2)(相对于Floyd算法来说) 是一种单源最短 ...

  10. 【XSY2666】排列问题 DP 容斥原理 分治FFT

    题目大意 有\(n\)种颜色的球,第\(i\)种有\(a_i\)个.设\(m=\sum a_i\).你要把这\(m\)个小球排成一排.有\(q\)个询问,每次给你一个\(x\),问你有多少种方案使得相 ...