Description

对于刚上大学的牛牛来说,他面临的第一个问题是如何根据实际情况申请合适的课程。

在可以选择的课程中,有 2n 节课程安排在n个时间段上。在第 i(1≤i≤n)个时间段上,两节内容相同的课程同时在不同的地点进行,其中,牛牛预先被安排在教室 ci 上课,而另一节课程在教室 di 进行。

在不提交任何申请的情况下,学生们需要按时间段的顺序依次完成所有的 n 节安排好的课程。如果学生想更换第 i 节课程的教室,则需要提出申请。若申请通过,学生就可以在第 i个时间段去教室 di 上课,否则仍然在教室 ci 上课。

由于更换教室的需求太多,申请不一定能获得通过。通过计算,牛牛发现申请更换第 i 节课程的教室时,申请被通过的概率是一个已知的实数 ki,并且对于不同课程的申请,被通过的概率是互相独立的。

学校规定,所有的申请只能在学期开始前一次性提交,并且每个人只能选择至多 节课程进行申请。这意味着牛牛必须一次性决定是否申请更换每节课的教室,而不能根据某些课程的申请结果来决定其他课程是否申请;牛牛可以申请自己最希望更换教室的 m 门课程,也可以不用完这 m 个申请的机会,甚至可以一门课程都不申请。

因为不同的课程可能会被安排在不同的教室进行,所以牛牛需要利用课间时间从一间教室赶到另一间教室。

牛牛所在的大学有 v 个教室,有 e 条道路。每条道路连接两间教室,并且是可以双向通行的。由于道路的长度和拥堵程度不同,通过不同的道路耗费的体力可能会有所不同。 当第 i(1≤i≤n−1)节课结束后,牛牛就会从这节课的教室出发,选择一条耗费体力最少的路径前往下一节课的教室。

现在牛牛想知道,申请哪几门课程可以使他因在教室间移动耗费的体力值的总和的期望值最小,请你帮他求出这个最小值。

Input

从标准输入读入数据。

第一行四个整数 n,m,v,e。n 表示这个学期内的时间段的数量;m 表示牛牛最多可以申请更换多少节课程的教室;v 表示牛牛学校里教室的数量;e表示牛牛的学校里道路的数量。

第二行 n 个正整数,第 (1≤i≤n)个正整数表示 ci,即第 个时间段牛牛被安排上课的教室;保证 1≤ci≤v

第三行 n 个正整数,第 i(1≤i≤n)个正整数表示 di,即第 i 个时间段另一间上同样课程的教室;保证 1≤di≤v。

第四行 n 个实数,第 i(1≤i≤n)个实数表示 ki,即牛牛申请在第 i 个时间段更换教室获得通过的概率。保证 0≤ki≤1。

接下来 e 行,每行三个正整数 aj,bj,wj,表示有一条双向道路连接教室 aj,bj,通过这条道路需要耗费的体力值是 w;保证 1≤aj,bj≤v, 1≤wj≤100。

保证 1≤n≤2000,0≤m≤2000,1≤v≤300,0≤e≤90000。

保证通过学校里的道路,从任何一间教室出发,都能到达其他所有的教室。

保证输入的实数最多包含 3 位小数。

Output

输出到标准输出。

输出一行,包含一个实数,四舍五入精确到小数点后恰好位,表示答案。你的输出必须和标准输出完全一样才算正确。

测试数据保证四舍五入后的答案和准确答案的差的绝对值不大于 4×10^{−3}。 (如果你不知道什么是浮点误差,这段话可以理解为:对于大多数的算法,你可以正常地使用浮点数类型而不用对它进行特殊的处理)

Sample Input

  1. 3 2 3 3
  2. 2 1 2
  3. 1 2 1
  4. 0.8 0.2 0.5
  5. 1 2 5
  6. 1 3 3
  7. 2 3 1

Sample Output

  1. 2.80

概率DP

dp[i][j][k]表示牛牛已经处理完了前i个时间段的课程,一共用了j次申请,k的取值为0或1表示处理第i个时间段的课程时耗费体力值的最小期望,是否使用了申请。

分情况讨论(方程很简单,最好自己手推)

1.第i个时间段没有用申请

\(dp[i][j][0]=min(dp[i-1][j][0]+dis[c[i-1]][c[i]]*1.0,min(dp[i][j][0],dp[i-1][j][1]+k[i-1]*dis[d[i-1]][c[i]]+(1.0-k[i-1])*dis[c[i-1]][c[i]]))\)

i一定没有更改,只用讨论i-1有没有改,更改有没有成功就可以了。

2.第i个时间段用申请(j>0):\(dp[i][j][1]=min(dp[i-1][j-1][0]+dis[c[i-1]][c[i]]*(1.0-k[i])+dis[c[i-1]][d[i]]*k[i],min(dp[i][j][1],dp[i-1][j-1][1]+dis[d[i-1]][c[i]]*(1.0-k[i])*k[i-1]+dis[d[i-1]][d[i]]*k[i]*k[i-1]+dis[c[i-1]][c[i]]*(1.0-k[i])*(1.0-k[i-1])+dis[c[i-1]][d[i]]*k[i]*(1.0-k[i-1])))\)

这个要讨论i有没有改,更改有没有成功,i-1有没有改,更改有没有成功。

其中的dis要求v个点互相的最短路,因为v<=300,弗洛伊德O(n^3)的时间复杂度跑得了。

求出最短路后直接套状转方程就可以了。

  1. #include<bits/stdc++.h>
  2. #define inf 0x7ffffff
  3. using namespace std;
  4. int n,m,v,e,c[2001],d[2001],dis[301][301],x,y,z;
  5. double dp[2001][2001][2],k[2001];
  6. int main()
  7. {
  8. // freopen("in.in","r",stdin);
  9. scanf("%d%d%d%d",&n,&m,&v,&e);
  10. for(int i=1;i<=n;i++)
  11. {
  12. scanf("%d",&c[i]);
  13. }
  14. for(int i=1;i<=n;i++)
  15. {
  16. scanf("%d",&d[i]);
  17. }
  18. for(int i=1;i<=n;i++)
  19. {
  20. scanf("%lf",&k[i]);
  21. }
  22. for(int i=1;i<=v;i++)
  23. {
  24. for(int j=1;j<i;j++)
  25. {
  26. dis[i][j]=dis[j][i]=inf;
  27. }
  28. }
  29. for(int i=1;i<=e;i++)
  30. {
  31. scanf("%d%d%d",&x,&y,&z);
  32. dis[x][y]=dis[y][x]=min(z,dis[x][y]);
  33. }
  34. for(int kkk=1;kkk<=v;kkk++)
  35. {
  36. for(int i=1;i<=v;i++)
  37. {
  38. for(int j=1;j<=v;j++)
  39. {
  40. if(dis[i][j]>dis[i][kkk]+dis[kkk][j])
  41. {
  42. dis[j][i]=dis[i][j]=dis[i][kkk]+dis[kkk][j];
  43. }
  44. }
  45. }
  46. }
  47. for(int i=1;i<=n;i++)
  48. {
  49. for(int j=0;j<=m;j++)
  50. {
  51. dp[i][j][0]=dp[i][j][1]=inf;
  52. }
  53. }
  54. dp[1][0][0]=dp[1][1][1]=0.0;
  55. for(int i=2;i<=n;i++)
  56. {
  57. for(int j=0;j<=min(m,i);j++)
  58. {
  59. dp[i][j][0]=min(dp[i-1][j][0]+dis[c[i-1]][c[i]]*1.0,min(dp[i][j][0],dp[i-1][j][1]+k[i-1]*dis[d[i-1]][c[i]]+(1.0-k[i-1])*dis[c[i-1]][c[i]]));
  60. if(j)
  61. {
  62. dp[i][j][1]=min(dp[i-1][j-1][0]+dis[c[i-1]][c[i]]*(1.0-k[i])+dis[c[i-1]][d[i]]*k[i],min(dp[i][j][1],dp[i-1][j-1][1]+dis[d[i-1]][c[i]]*(1.0-k[i])*k[i-1]+dis[d[i-1]][d[i]]*k[i]*k[i-1]+dis[c[i-1]][c[i]]*(1.0-k[i])*(1.0-k[i-1])+dis[c[i-1]][d[i]]*k[i]*(1.0-k[i-1])));
  63. }
  64. }
  65. }
  66. double ans=inf;
  67. for(int i=0;i<=m;i++)
  68. {
  69. ans=min(ans,min(dp[n][i][0],dp[n][i][1]));
  70. }
  71. printf("%.2lf",ans);
  72. return 0;
  73. }

【BZOJ4720】【UOJ262】【NOIP2016】换教室的更多相关文章

  1. 【bzoj4720】[NOIP2016]换教室

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

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

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

  3. 【bzoj4720】[Noip2016]换教室 期望dp+最短路

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

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

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

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

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

  6. BZOJ 4720 [Noip2016]换教室

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

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

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

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

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

  9. [NOIp2016] 换教室

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

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

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

随机推荐

  1. git clone remote: HTTP Basic: Access denied

    git clone 项目失败,报下面的错误信息: $ git clone http://192.168.0.141/xxxx.git Cloning into 'appEnterprise'... r ...

  2. 织梦cms列表页获取标签

    <!-- 标签 --> [field:id runphp='yes'] global $cfg_cmspath; $tags = GetTags(@me); $revalue = ''; ...

  3. 使用.Htaccess文件实现301重定向常用的七种方法

    使用.Htaccess文件实现301重定向常用的七种方法   301重定向对广大站长来说并不陌生,从网站建设到目录优化,避免不了对网站目录进行更改,在这种情况下用户的收藏夹里面和搜索引擎里面可能保存的 ...

  4. 运算符 字符串 for循环

    1. 运算符 1.1赋值运算符 = += -= *= /= //= %= **= 1.2比较运算符 < > = <= == != 1.3成员运算符 in not in 1.4逻辑运算 ...

  5. netty源码解解析(4.0)-23 ByteBuf内存管理:分配和释放

    ByteBuf内存分配和释放由具体实现负责,抽象类型只定义的内存分配和释放的时机. 内存分配分两个阶段: 第一阶段,初始化时分配内存.第二阶段: 内存不够用时分配新的内存.ByteBuf抽象层没有定义 ...

  6. JSON说明

    1. JSON 数据的书写格式 对象:是一个无序的“‘名称/值’对”集合.一个对象以“{”(左括号)开始,“}”(右括号)结束.每个“名称”后跟一个“:”(冒号):“‘名称/值’ 对”之间使用“,”( ...

  7. 服务器配置https协议,三种免费的方法

    最近想搞一个网站玩玩,发布网站用https协议已经是大势所趋了.例如微信小程序,不使用https协议根本不让接入.所以,分享一下我尝试过的三种方法. 1.Linux自签(OPENSSL生成SSL自签证 ...

  8. springboot配置ehcache2.X缓存(@Cacheable等注解和手动操作缓存的工具类 支持element粒度的时间设置)

    本文只写出一些注意事项和源码,请善用官方文档~ 注解实现 @Cacheable @CachePut @CacheEvit 启动类上加@EnableCaching就可以开启缓存 由文档可知,自动检测缓存 ...

  9. Java集合框架,你了解多少?相信你看了这篇汇总一目了然!

    相信大多数的程序员都知道,Dictionary.Vertor.Stack和Properties这些类被用来存储和操作对象组.但是他们缺少一个核心的主题的. 集合框架设计成要满足以下的几个目标 第一条: ...

  10. 快速入门Maven(二)(Eclipse构建Maven项目)

    Mars2的eclipse()已经集成了Maven插件,所以用这个版本不需要装插件了. 接下来构建: 一.调整Eclipse设置 1.选择3.3.9版本的maven软件 2.修改默认的本地仓库地址 二 ...