[CF1067D]Computer Game[凸包/斜率优化+倍增+矩阵乘法]
题意
你有 \(n\) 个任务,初始收益为 \(a\) ,共 \(t\) 轮游戏,每轮可以选择完成一个任务(可以做多次),完成之后可以给任意任务升级,升级之后的任务收益为 \(b\) ,每个任务还有完成的概率 \(p\) ,问期望最大收益是多少。
\(n\leq 10^5,1\leq a< b\leq 10^8,t\leq 10^9\)
分析
一个重要而显然的结论是如果我们有了一次升级的机会,一定升级 \(b*p\) 最大的那一个,之后一直选择完成这个任务,记 \(M\) 表示最大的 \(b*p\) 。然后只关注没有一次完成时的最大期望收益。
定义状态 \(f_t\) 表示还剩 \(t\) 轮游戏,还没有完成一个任务的期望最大收益。
转移: \(f_{t+1}=\max\{p_i*(tM+a_i)+(1-p_i)*f_t\}\).
可以变形成斜率优化的形式(注意这里的 \(f_t\) 要看成 \(k\) 的一部分) ,也可以写成这种形式:
\]
将 \(p_i\) 看成 \(k\) , \(p_ia_i\) 看成 \(b\) ,然后每个转移都是形如直线 \(y=kx+b\) 的形式。
我们可以维护一个下凸壳,每次可以二分当前的 \(x=tM-f_t\) 在哪一段,可以做到 \(Tlogn\) 的时间。
发现 \(tM-f_t\)是单调不降的,证明如下:
设 \(x_{t+1} \geq x_t\)
有:\(tM-f_t\geq (t-1)M-f_{t-1}\)
\(M\geq f_t-f_{t-1}\)
考虑 \(f\) 的实际意义可以发现上式一定成立,因为一轮的期望收益不会超过 \(M\) 。
可以考虑构造,对于 \(t+1\) 轮的最优决策, \(t\) 轮可以复制过来(除开第一轮),而在 \(t+1\) 轮产生的每种情况,\(t\) 轮都可以效仿(发生的概率是相同的),发现期望收益只有最后一轮不一样,最多差为 \(M\)。
然后在每一段倍增矩乘即可。
总时间复杂度为 \(O(n\log T)\)。
代码
#include<bits/stdc++.h>
using namespace std;
#define go(u) for(int i=head[u],v=e[i].to;i;i=e[i].lst,v=e[i].to)
#define rep(i,a,b) for(int i=a;i<=b;++i)
#define pb push_back
typedef long long LL;
inline int gi(){
int x=0,f=1;char ch=getchar();
while(!isdigit(ch)) {if(ch=='-') f=-1;ch=getchar();}
while(isdigit(ch)){x=(x<<3)+(x<<1)+ch-48;ch=getchar();}
return x*f;
}
template<typename T>inline bool Max(T &a,T b){return a<b?a=b,1:0;}
template<typename T>inline bool Min(T &a,T b){return b<a?a=b,1:0;}
const int N=1e5 + 7;
typedef double db;
const db eps=1e-12;
int n,len,tp,st[N];
LL t;
db a[N],b[N],p[N],M;
int dcmp(db x){
if(fabs(x)<eps) return 0;
return x<0?-1:1;
}
struct line{
db k,b;int id;
line(){}line(db k,db b,int id):k(k),b(b),id(id){}
bool operator <(const line &rhs)const{
if(dcmp(k-rhs.k)!=0) return dcmp(k-rhs.k)<0;
return dcmp(b-rhs.b)<0;
}
}l[N];
db X(int a,int b){
return (l[b].b-l[a].b)/(l[a].k-l[b].k);
}
struct mat{
db v[5][5];
mat(){memset(v,0,sizeof v);}
void init(){memset(v,0,sizeof v);}
mat operator *(const mat &rhs)const{
mat res;
rep(i,0,4)
rep(j,0,4)
rep(k,0,4)
res.v[i][j]+=v[i][k]*rhs.v[k][j];
return res;
}
}A,B[34],C;
mat Pow(mat a,LL b){
mat res;
rep(i,1,4) res.v[i][i]=1;
for(;b;b>>=1,a=a*a) if(b&1) res=res*a;
return res;
}
int main(){
scanf("%d%I64d\n",&n,&t);
rep(i,1,n) {
scanf("%lf%lf%lf",&a[i],&b[i],&p[i]);
l[i]=line(p[i],p[i]*a[i],i);
Max(M,b[i]*p[i]);
}
sort(l+1,l+1+n);
rep(i,1,n) {
if(dcmp(l[i].k-l[i+1].k)==0) continue;
l[++len]=l[i];
}
rep(i,1,len){
while(tp>1&&dcmp(X(st[tp-1],i)-X(st[tp],st[tp-1]))<=0) --tp;
st[++tp]=i;
}
int now=1;LL cnt=0;
for(int now=1;now<=tp&&cnt^t;++now){
double R=cnt*M-A.v[1][1];
while(now<tp&&X(st[now],st[now+1])<=R) ++now;
int i=st[now];R=X(st[now],st[now+1]);
B[0].init();
B[0].v[1][1]=1-l[i].k;
B[0].v[2][1]=l[i].k;
B[0].v[2][2]=B[0].v[3][1]=B[0].v[3][3]=B[0].v[4][2]=B[0].v[4][4]=1;
A.v[1][3]=l[i].b,A.v[1][4]=M;
rep(i,1,33) B[i]=B[i-1]*B[i-1];
for(int i=33;~i;--i)if(t-cnt>(1ll<<i)){
C=A*B[i];
if(now==tp||dcmp((cnt+(1ll<<i))*M-C.v[1][1]-R)<0) A=C,cnt+=(1ll<<i);
}
cnt++,A=A*B[0];
}
printf("%.13lf\n",A.v[1][1]);
return 0;
}
[CF1067D]Computer Game[凸包/斜率优化+倍增+矩阵乘法]的更多相关文章
- CF1067D. Computer Game(斜率优化+倍增+矩阵乘法)
题目链接 https://codeforces.com/contest/1067/problem/D 题解 首先,如果我们获得了一次升级机会,我们一定希望升级 \(b_i \times p_i\) 最 ...
- CF781D Axel and Marston in Bitland [倍增 矩阵乘法 bitset]
Axel and Marston in Bitland 好开心第一次补$F$题虽然是$Div.2$ 题意: 一个有向图,每条边是$0$或$1$,要求按如下规则构造一个序列然后走: 第一个是$0$,每次 ...
- 【loj2325】「清华集训 2017」小Y和恐怖的奴隶主 概率dp+倍增+矩阵乘法
题目描述 你有一个m点生命值的奴隶主,奴隶主受伤未死且当前随从数目不超过k则再召唤一个m点生命值的奴隶主. T次询问,每次询问如果如果对面下出一个n点攻击力的克苏恩,你的英雄期望会受到到多少伤害. 输 ...
- 倍增&矩阵乘法 专题复习
倍增&矩阵乘法 专题复习 PreWords 这两个基础算法我就不多说啦,但是还是要介绍一下" 广义矩阵 "乘法 其实就是把矩阵换成取\(max\),然后都一样... 据神仙 ...
- BZOJ 3672[NOI2014]购票(树链剖分+线段树维护凸包+斜率优化) + BZOJ 2402 陶陶的难题II (树链剖分+线段树维护凸包+分数规划+斜率优化)
前言 刚开始看着两道题感觉头皮发麻,后来看看题解,发现挺好理解,只是代码有点长. BZOJ 3672[NOI2014]购票 中文题面,题意略: BZOJ 3672[NOI2014]购票 设f(i)f( ...
- 4.28 省选模拟赛 负环 倍增 矩阵乘法 dp
容易想到 这个环一定是简单环. 考虑如果是复杂环 那么显然对于其中的第一个简单环来说 要么其权值为负 如果为正没必要走一圈 走一部分即可. 对于前者 显然可以找到更小的 对于第二部分是递归定义的. 综 ...
- Codeforces 576D - Flights for Regular Customers(bitset 优化广义矩阵乘法)
题面传送门 题意: 有一张 \(n\) 个点 \(m\) 条边的有向图,你初始在 \(1\) 号点,边上有边权 \(c_i\) 表示只有当你经过至少 \(c_i\) 条边的时候你才能经过第 \(i\) ...
- 【POJ3613】Cow Relays 离散化+倍增+矩阵乘法
题目大意:给定一个 N 个顶点,M 条边的无向图,求从起点到终点恰好经过 K 个点的最短路. 题解:设 \(d[1][i][j]\) 表示恰好经过一条边 i,j 两点的最短路,那么有 \(d[r+m] ...
- dp斜率优化
算法-dp斜率优化 前置知识: 凸包 斜率优化很玄学,凭空讲怎么也讲不好,所以放例题. [APIO2014]序列分割 [APIO2014]序列分割 给你一个长度为 \(n\) 的序列 \(a_1,a_ ...
随机推荐
- CSS样式----CSS样式表的继承性和层叠性(图文详解)
本文最初于2017-07-29发表于博客园,并在GitHub上持续更新前端的系列文章.欢迎在GitHub上关注我,一起入门和进阶前端. 以下是正文. 本文重点 CSS的继承性 CSS的层叠性 计算权重 ...
- [原创]RedHat 安装MySQL数据库
朋友购买了阿里云的服务器,服务器上自带有CentOS操作系统,但是开发软件需要自己安装,接下来将介绍本地RedHat Linux 5.10虚拟机上搭建Mysql数据库. 一.软件准备 (1)jdk-6 ...
- Linux清除用户登录记录和命令历史方法(个人笔记)
清除登陆系统成功的记录 [root@localhost root]# echo > /var/log/wtmp //此文件默认打开时乱码,可查到ip等信息 [root@localhost roo ...
- October 23rd, 2017 Week 43rd Monday
Champions have the courage to keep turning the pages because they know a better chapter lies ahead. ...
- 【Ansible 文档】【译文】常见问题
http://docs.ansible.com/ansible/latest/faq.html 如何为一个task或者整个Playbook设置PATH或者任意其他环境变量? 通过environment ...
- python面试题(一)
1.通过代码实现如下转换: 二进制转换成十进制:v = “0b1111011” #先将其转换为字符串,再使用int函数,指定进制转换为十进制. print(int("0b1111011&qu ...
- SQL操作语句
SQL语句与Mysql的语句大体上比较相似.以下是sql server的一套练习题,是很好的数据库操作语句学习资料,学校的学习资料,在此整理了以下. 数据库exam:这是一个模拟电子商务,网上直销的数 ...
- js之上传文件多图片预览
多图片上传预览功能也是现在非常常用的 下面是html代码: <html xmlns="http://www.w3.org/1999/xhtml"> <head&g ...
- AOP实现拦截对象以及获取切入目标方法和注解
AOP实现拦截对象以及获取切入目标方法和注解 一.JoinPoint是什么? AspectJ使用org.aspectj.lang.JoinPoint接口表示目标类连接点对象,如果是环绕增强时,使用 o ...
- leetcode16—3 Sum Closet
Given an array nums of n integers and an integer target, find three integers in nums such that the s ...