[计蒜客] 矿石采集【记搜、Tarjan缩点+期望Dp】
Online Judge:计蒜客信息学3月提高组模拟赛
Label:记搜,TarJan缩点,树状数组,期望Dp
题解
整个题目由毫无关联的两个问题组合成:
part1
问题:对于每个询问的起点终点,求出起点到终点能够收获的最大矿石价值。
思路:
1.矿石种类不多,且收获价值时只与是否拥有有关,与收集数量无关,可以用状态压缩表示当前收集矿石的状态;
2.由于是单向边且可能形成环,用\(Tarjan\)缩点,然后进行转移。或者直接记忆化dfs/bfs,用\(dp[i][j][sta]\)表示,从\(i\)到\(j\),是否有可能收集到状态为\(sta\)的矿石。最后求一遍最大值即可。
part2
问题:安排执行哪些委托,使得期望获利最大。
思路:
1.整理委托的信息:
对于一份委托\((yi,si,ti,pi,ci,li)\),我们可以根据\((si,ti)\)以及前一个部分的预处理来计算出这份委托的收益(还没减去成本)\(wi\);这份委托的开始时间和结束时间\((yi,yi+li)\),这里简记为\((l,r)\)。这样我们可以用一个结构体表示一份委托{\(l,r,w,c,p\)}。
2.确定转移顺序
毫无疑问先根据开始时间\(li\)排序所有委托。
定义状态\(dp[i]\),表示已经考虑了\(i..q\)的委托,且接受了委托\(i\)时的最大期望获利。接着只用倒序转移即可。对于当前的委托\(i\),我们利用二分查找在\(O(logq)\)的时间内,找到第一个开始时间大于等于当前委托结束时间的委托\(j\)。然后转移方程为:\(dp[i]=max(dp[j...q]+w_i)*(1-p_i)-c_i\),这个很好理解就不解释了,而区间\([j,q]\)的最大值可以用树状数组/线段树维护。
这题就结束了,主要难点是题目变量比较多,只要理清关系分成两个问题求解就好了。
part1的做法比较多,不过由于数据范围小都可行。part2的时间复杂度为\(O(NlogN)\)。
#include<bits/stdc++.h>
using namespace std;
const int N=110,M=210,Q=1e5+10;
int val[1030];
int n,m,k,q,has[N];
struct edge{int to,nxt;}e[M];
int head[N],cnt;
inline void link(int u,int v){
e[++cnt].to=v;e[cnt].nxt=head[u];
head[u]=cnt;
}
bool ok[N][N][1030];
void dfs(int s,int x,int sta){
if(ok[s][x][sta])return;
ok[s][x][sta]=1;
for(int i=head[x];i;i=e[i].nxt){
int y=e[i].to;dfs(s,y,sta|has[y]);
}
}
struct Node{
int l,r,w,c;
double p;
}x[Q];
inline bool cmp(Node a,Node b){return a.l<b.l;}
void pre(){
for(register int i=1;i<=n;i++)dfs(i,i,has[i]);
for(register int i=1,u,v;i<=q;i++){
scanf("%d%d%d%lf%d%d",&x[i].l,&u,&v,&x[i].p,&x[i].c,&x[i].r);
x[i].r+=x[i].l;
for(register int o=0;o<(1<<k);o++)if(ok[u][v][o]){
if(val[o]>x[i].w)x[i].w=val[o];
}
}
}
double c[Q];
inline double ask(int x){
double res=0;
while(x<=q){res=max(res,c[x]);x+=x&(-x);}
return res;
}
inline void update(int x,double d){
while(x){c[x]=max(c[x],d);x-=x&(-x);}
}
inline void Dp(){
sort(x+1,x+q+1,cmp);
for(register int i=q;i>=1;i--){
//如果接受i委托,下一个最早能进行的委托是id
int l=i+1,r=q,id=q+1;
while(l<=r){
int mid=l+r>>1;
if(x[i].r<=x[mid].l)r=mid-1,id=mid;
else l=mid+1;
}
double now=(ask(id)+x[i].w)*(1.0-x[i].p)-x[i].c;
update(i,now);
}
printf("%.6f\n",ask(1));
}
int main(){
scanf("%d%d%d%d",&n,&m,&k,&q);
for(register int i=1;i<=n;i++){
char s[12];scanf("%s",s);
for(int j=0;j<strlen(s);j++)if(s[j]=='1')has[i]|=1<<j;
}
for(register int i=1;i<=m;i++){
int u,v;scanf("%d%d",&u,&v);
link(u,v);
}
for(register int i=0,w;i<k;i++){
scanf("%d",&w);
for(int j=0;j<(1<<k);j++)if((1<<i)&j)val[j]+=w;
}
pre();
Dp();
}
[计蒜客] 矿石采集【记搜、Tarjan缩点+期望Dp】的更多相关文章
- 计蒜客 NOIP 提高组模拟竞赛第一试 补记
计蒜客 NOIP 提高组模拟竞赛第一试 补记 A. 广场车神 题目大意: 一个\(n\times m(n,m\le2000)\)的网格,初始时位于左下角的\((1,1)\)处,终点在右上角的\((n, ...
- 计蒜客 作弊揭发者(string的应用)
鉴于我市拥堵的交通状况,市政交管部门经过听证决定在道路两侧安置自动停车收费系统.当车辆驶入车位,系统会通过配有的摄像头拍摄车辆画面,通过识别车牌上的数字.字母序列识别车牌,通过连接车管所车辆信息数据库 ...
- 计蒜客的一道题dfs
这是我无聊时在计蒜客发现的一道题. 题意: 蒜头君有一天闲来无事和小萌一起玩游戏,游戏的内容是这样的:他们不知道从哪里找到了N根不同长度的木棍, 看谁能猜出这些木棍一共能拼出多少个不同的不等边三角形. ...
- 计蒜客模拟赛5 D2T1 成绩统计
又到了一年一度的新生入学季了,清华和北大的计算机系同学都参加了同一场开学考试(因为两校兄弟情谊深厚嘛,来一场联考还是很正常的). 不幸的是,正当老师要统计大家的成绩时,世界上的所有计算机全部瘫痪了. ...
- 计蒜客 等边三角形 dfs
题目: https://www.jisuanke.com/course/2291/182238 思路: 1.dfs(int a,int b,int c,int index)//a,b,c三条边的边长, ...
- 计蒜客 方程的解数 dfs
题目: https://www.jisuanke.com/course/2291/182237 思路: 来自:https://blog.csdn.net/qq_29980371/article/det ...
- 计蒜客 买书 dfs
题目: https://www.jisuanke.com/course/2291/182236 思路: 递归解决,从第一本书开始,每本书都有两种选择: //index是book里面每本书价格的下标, ...
- 计蒜客:Entertainment Box
Ada, Bertrand and Charles often argue over which TV shows to watch, and to avoid some of their fight ...
- 爬虫acm比赛成绩(多页成绩整合在一起、获取复制不了的数据)(hihocoder、计蒜客)
https://github.com/congmingyige/web-crawler_rank-of-competition-in-JiSuanKe-and-hihocoder 1. 计蒜客(获取复 ...
随机推荐
- vue项目 上传到git
1. git add . 效果: 2. git commit -m '' 3. git push 效果: 4.在码云上就可以看到修改啦 2018-05-19 08:52:52
- 【JZOJ6357】小ω的图(graph)
description analysis 拆位从高位到低位贪心 对于当前位,如果把所有当前位为\(1\)的边塞入,\(1\)和\(n\)连通,则该位必须为\(1\) 这个是因为高位的\(1\)比所有低 ...
- 阿里云POLARDB如何帮助猿辅导打造“孩子喜欢老师好”的网课平台?
海量的题库.音视频答题资料.用户数据以及日志,对猿辅导后台数据存储和处理能力都提出了严峻的要求.而由于教育辅导行业的业务特点,猿辅导也面临着业务峰值对于数据库能力的巨大挑战.本文就为大家介绍阿里云PO ...
- CSS-基本语法/引用/文本设置/选择器/css3属性
CSS-基本语法/引用/文本设置 css基本语法及页面引用 css基本语法 css的定义方法是: 选择器 { 属性:值; 属性:值; 属性:值;} 选择器是将样式和页面元素关联起来的名称,属性是希望设 ...
- thinkphp 链接数据库
ThinkPHP内置了抽象数据库访问层,把不同的数据库操作封装起来,我们只需要使用公共的Db类进行操作,而无需针对不同的数据库写不同的代码和底层实现,Db类会自动调用相应的数据库驱动来处理.目前的数据 ...
- SQL Server [join] 整理
[表A]Aid Aname Acode1 aa 0012 bb 0023 cc 0034 dd 004 ...
- (转)iframe 高度100%时,出现垂直滚动条
问题 需求是这样的,iframe在一个div中,并且iframe高度与div一样,所以设置了iframe高度是100%,结果div出现了滚动条,在排除了padding.margin的因素外,还是有滚动 ...
- myeclipse CTRL+1功能
有时候,在myeclipse或者eclipse中自动编译代码有错误,我们把鼠标放在错误一行能够自动显示出问题原因,但是有时显示问题让人有些匪夷所思,不知所云何物. 此时可以使用<ctrl> ...
- Vue项目组织规范
目录 一.项目结构的核心思想 二.项目目录结构 三.资源路径编译规则 四.index.html 五.build目录 和 config目录 六.public目录 七.static 目录 八.src目录结 ...
- day 56 Django基础五之django模型层(二)多表操作
Django基础五之django模型层(二)多表操作 本节目录 一 创建模型 二 添加表记录 三 基于对象的跨表查询 四 基于双下划线的跨表查询 五 聚合查询.分组查询.F查询和Q查询 六 ORM ...